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.
Files changed (190) hide show
  1. c63a5cfe_b235_4fbe_8bbb_82a9e02a482a_python-0.1.0a3.dist-info/METADATA +396 -0
  2. c63a5cfe_b235_4fbe_8bbb_82a9e02a482a_python-0.1.0a3.dist-info/RECORD +190 -0
  3. c63a5cfe_b235_4fbe_8bbb_82a9e02a482a_python-0.1.0a3.dist-info/WHEEL +4 -0
  4. c63a5cfe_b235_4fbe_8bbb_82a9e02a482a_python-0.1.0a3.dist-info/licenses/LICENSE +201 -0
  5. digitalocean_genai_sdk/__init__.py +99 -0
  6. digitalocean_genai_sdk/_base_client.py +1949 -0
  7. digitalocean_genai_sdk/_client.py +795 -0
  8. digitalocean_genai_sdk/_compat.py +219 -0
  9. digitalocean_genai_sdk/_constants.py +14 -0
  10. digitalocean_genai_sdk/_exceptions.py +108 -0
  11. digitalocean_genai_sdk/_files.py +123 -0
  12. digitalocean_genai_sdk/_models.py +805 -0
  13. digitalocean_genai_sdk/_qs.py +150 -0
  14. digitalocean_genai_sdk/_resource.py +43 -0
  15. digitalocean_genai_sdk/_response.py +832 -0
  16. digitalocean_genai_sdk/_streaming.py +333 -0
  17. digitalocean_genai_sdk/_types.py +219 -0
  18. digitalocean_genai_sdk/_utils/__init__.py +57 -0
  19. digitalocean_genai_sdk/_utils/_logs.py +25 -0
  20. digitalocean_genai_sdk/_utils/_proxy.py +65 -0
  21. digitalocean_genai_sdk/_utils/_reflection.py +42 -0
  22. digitalocean_genai_sdk/_utils/_resources_proxy.py +24 -0
  23. digitalocean_genai_sdk/_utils/_streams.py +12 -0
  24. digitalocean_genai_sdk/_utils/_sync.py +86 -0
  25. digitalocean_genai_sdk/_utils/_transform.py +447 -0
  26. digitalocean_genai_sdk/_utils/_typing.py +151 -0
  27. digitalocean_genai_sdk/_utils/_utils.py +422 -0
  28. digitalocean_genai_sdk/_version.py +4 -0
  29. digitalocean_genai_sdk/lib/.keep +4 -0
  30. digitalocean_genai_sdk/py.typed +0 -0
  31. digitalocean_genai_sdk/resources/__init__.py +145 -0
  32. digitalocean_genai_sdk/resources/agents/__init__.py +89 -0
  33. digitalocean_genai_sdk/resources/agents/agents.py +965 -0
  34. digitalocean_genai_sdk/resources/agents/api_keys.py +581 -0
  35. digitalocean_genai_sdk/resources/agents/child_agents.py +508 -0
  36. digitalocean_genai_sdk/resources/agents/functions.py +421 -0
  37. digitalocean_genai_sdk/resources/agents/knowledge_bases.py +346 -0
  38. digitalocean_genai_sdk/resources/agents/versions.py +298 -0
  39. digitalocean_genai_sdk/resources/api_keys/__init__.py +19 -0
  40. digitalocean_genai_sdk/resources/api_keys/api_keys.py +275 -0
  41. digitalocean_genai_sdk/resources/api_keys/api_keys_.py +529 -0
  42. digitalocean_genai_sdk/resources/auth/__init__.py +33 -0
  43. digitalocean_genai_sdk/resources/auth/agents/__init__.py +33 -0
  44. digitalocean_genai_sdk/resources/auth/agents/agents.py +102 -0
  45. digitalocean_genai_sdk/resources/auth/agents/token.py +173 -0
  46. digitalocean_genai_sdk/resources/auth/auth.py +102 -0
  47. digitalocean_genai_sdk/resources/chat.py +381 -0
  48. digitalocean_genai_sdk/resources/embeddings.py +201 -0
  49. digitalocean_genai_sdk/resources/indexing_jobs.py +543 -0
  50. digitalocean_genai_sdk/resources/knowledge_bases/__init__.py +33 -0
  51. digitalocean_genai_sdk/resources/knowledge_bases/data_sources.py +410 -0
  52. digitalocean_genai_sdk/resources/knowledge_bases/knowledge_bases.py +667 -0
  53. digitalocean_genai_sdk/resources/models.py +222 -0
  54. digitalocean_genai_sdk/resources/providers/__init__.py +47 -0
  55. digitalocean_genai_sdk/resources/providers/anthropic/__init__.py +33 -0
  56. digitalocean_genai_sdk/resources/providers/anthropic/anthropic.py +102 -0
  57. digitalocean_genai_sdk/resources/providers/anthropic/keys.py +662 -0
  58. digitalocean_genai_sdk/resources/providers/openai/__init__.py +33 -0
  59. digitalocean_genai_sdk/resources/providers/openai/keys.py +658 -0
  60. digitalocean_genai_sdk/resources/providers/openai/openai.py +102 -0
  61. digitalocean_genai_sdk/resources/providers/providers.py +134 -0
  62. digitalocean_genai_sdk/resources/regions.py +191 -0
  63. digitalocean_genai_sdk/types/__init__.py +57 -0
  64. digitalocean_genai_sdk/types/agent_create_params.py +39 -0
  65. digitalocean_genai_sdk/types/agent_create_response.py +16 -0
  66. digitalocean_genai_sdk/types/agent_delete_response.py +16 -0
  67. digitalocean_genai_sdk/types/agent_list_params.py +18 -0
  68. digitalocean_genai_sdk/types/agent_list_response.py +198 -0
  69. digitalocean_genai_sdk/types/agent_retrieve_response.py +16 -0
  70. digitalocean_genai_sdk/types/agent_update_params.py +65 -0
  71. digitalocean_genai_sdk/types/agent_update_response.py +16 -0
  72. digitalocean_genai_sdk/types/agent_update_status_params.py +16 -0
  73. digitalocean_genai_sdk/types/agent_update_status_response.py +16 -0
  74. digitalocean_genai_sdk/types/agents/__init__.py +31 -0
  75. digitalocean_genai_sdk/types/agents/api_key_create_params.py +15 -0
  76. digitalocean_genai_sdk/types/agents/api_key_create_response.py +12 -0
  77. digitalocean_genai_sdk/types/agents/api_key_delete_response.py +12 -0
  78. digitalocean_genai_sdk/types/agents/api_key_list_params.py +15 -0
  79. digitalocean_genai_sdk/types/agents/api_key_list_response.py +18 -0
  80. digitalocean_genai_sdk/types/agents/api_key_regenerate_response.py +12 -0
  81. digitalocean_genai_sdk/types/agents/api_key_update_params.py +19 -0
  82. digitalocean_genai_sdk/types/agents/api_key_update_response.py +12 -0
  83. digitalocean_genai_sdk/types/agents/api_link_knowledge_base_output.py +16 -0
  84. digitalocean_genai_sdk/types/agents/api_links.py +21 -0
  85. digitalocean_genai_sdk/types/agents/api_meta.py +15 -0
  86. digitalocean_genai_sdk/types/agents/child_agent_add_params.py +22 -0
  87. digitalocean_genai_sdk/types/agents/child_agent_add_response.py +14 -0
  88. digitalocean_genai_sdk/types/agents/child_agent_delete_response.py +13 -0
  89. digitalocean_genai_sdk/types/agents/child_agent_update_params.py +24 -0
  90. digitalocean_genai_sdk/types/agents/child_agent_update_response.py +18 -0
  91. digitalocean_genai_sdk/types/agents/child_agent_view_response.py +16 -0
  92. digitalocean_genai_sdk/types/agents/function_create_params.py +25 -0
  93. digitalocean_genai_sdk/types/agents/function_create_response.py +16 -0
  94. digitalocean_genai_sdk/types/agents/function_delete_response.py +16 -0
  95. digitalocean_genai_sdk/types/agents/function_update_params.py +29 -0
  96. digitalocean_genai_sdk/types/agents/function_update_response.py +16 -0
  97. digitalocean_genai_sdk/types/agents/knowledge_base_detach_response.py +16 -0
  98. digitalocean_genai_sdk/types/agents/version_list_params.py +15 -0
  99. digitalocean_genai_sdk/types/agents/version_list_response.py +118 -0
  100. digitalocean_genai_sdk/types/agents/version_update_params.py +15 -0
  101. digitalocean_genai_sdk/types/agents/version_update_response.py +30 -0
  102. digitalocean_genai_sdk/types/api_agent.py +263 -0
  103. digitalocean_genai_sdk/types/api_agent_api_key_info.py +22 -0
  104. digitalocean_genai_sdk/types/api_agreement.py +17 -0
  105. digitalocean_genai_sdk/types/api_anthropic_api_key_info.py +22 -0
  106. digitalocean_genai_sdk/types/api_deployment_visibility.py +9 -0
  107. digitalocean_genai_sdk/types/api_indexing_job.py +43 -0
  108. digitalocean_genai_sdk/types/api_key_list_params.py +42 -0
  109. digitalocean_genai_sdk/types/api_key_list_response.py +42 -0
  110. digitalocean_genai_sdk/types/api_keys/__init__.py +13 -0
  111. digitalocean_genai_sdk/types/api_keys/api_key_create_params.py +11 -0
  112. digitalocean_genai_sdk/types/api_keys/api_key_create_response.py +12 -0
  113. digitalocean_genai_sdk/types/api_keys/api_key_delete_response.py +12 -0
  114. digitalocean_genai_sdk/types/api_keys/api_key_list_params.py +15 -0
  115. digitalocean_genai_sdk/types/api_keys/api_key_list_response.py +18 -0
  116. digitalocean_genai_sdk/types/api_keys/api_key_update_params.py +15 -0
  117. digitalocean_genai_sdk/types/api_keys/api_key_update_regenerate_response.py +12 -0
  118. digitalocean_genai_sdk/types/api_keys/api_key_update_response.py +12 -0
  119. digitalocean_genai_sdk/types/api_keys/api_model_api_key_info.py +22 -0
  120. digitalocean_genai_sdk/types/api_knowledge_base.py +37 -0
  121. digitalocean_genai_sdk/types/api_model.py +57 -0
  122. digitalocean_genai_sdk/types/api_model_version.py +15 -0
  123. digitalocean_genai_sdk/types/api_openai_api_key_info.py +25 -0
  124. digitalocean_genai_sdk/types/api_retrieval_method.py +13 -0
  125. digitalocean_genai_sdk/types/auth/__init__.py +3 -0
  126. digitalocean_genai_sdk/types/auth/agents/__init__.py +6 -0
  127. digitalocean_genai_sdk/types/auth/agents/token_create_params.py +13 -0
  128. digitalocean_genai_sdk/types/auth/agents/token_create_response.py +13 -0
  129. digitalocean_genai_sdk/types/chat_completion_request_message_content_part_text_param.py +15 -0
  130. digitalocean_genai_sdk/types/chat_completion_token_logprob.py +57 -0
  131. digitalocean_genai_sdk/types/chat_create_completion_params.py +208 -0
  132. digitalocean_genai_sdk/types/chat_create_completion_response.py +81 -0
  133. digitalocean_genai_sdk/types/embedding_create_params.py +28 -0
  134. digitalocean_genai_sdk/types/embedding_create_response.py +41 -0
  135. digitalocean_genai_sdk/types/indexing_job_create_params.py +14 -0
  136. digitalocean_genai_sdk/types/indexing_job_create_response.py +12 -0
  137. digitalocean_genai_sdk/types/indexing_job_list_params.py +15 -0
  138. digitalocean_genai_sdk/types/indexing_job_list_response.py +18 -0
  139. digitalocean_genai_sdk/types/indexing_job_retrieve_data_sources_response.py +52 -0
  140. digitalocean_genai_sdk/types/indexing_job_retrieve_response.py +12 -0
  141. digitalocean_genai_sdk/types/indexing_job_update_cancel_params.py +14 -0
  142. digitalocean_genai_sdk/types/indexing_job_update_cancel_response.py +12 -0
  143. digitalocean_genai_sdk/types/knowledge_base_create_params.py +64 -0
  144. digitalocean_genai_sdk/types/knowledge_base_create_response.py +12 -0
  145. digitalocean_genai_sdk/types/knowledge_base_delete_response.py +11 -0
  146. digitalocean_genai_sdk/types/knowledge_base_list_params.py +15 -0
  147. digitalocean_genai_sdk/types/knowledge_base_list_response.py +18 -0
  148. digitalocean_genai_sdk/types/knowledge_base_retrieve_response.py +30 -0
  149. digitalocean_genai_sdk/types/knowledge_base_update_params.py +27 -0
  150. digitalocean_genai_sdk/types/knowledge_base_update_response.py +12 -0
  151. digitalocean_genai_sdk/types/knowledge_bases/__init__.py +16 -0
  152. digitalocean_genai_sdk/types/knowledge_bases/api_file_upload_data_source.py +15 -0
  153. digitalocean_genai_sdk/types/knowledge_bases/api_file_upload_data_source_param.py +15 -0
  154. digitalocean_genai_sdk/types/knowledge_bases/api_knowledge_base_data_source.py +35 -0
  155. digitalocean_genai_sdk/types/knowledge_bases/api_spaces_data_source.py +15 -0
  156. digitalocean_genai_sdk/types/knowledge_bases/api_spaces_data_source_param.py +15 -0
  157. digitalocean_genai_sdk/types/knowledge_bases/api_web_crawler_data_source.py +26 -0
  158. digitalocean_genai_sdk/types/knowledge_bases/api_web_crawler_data_source_param.py +25 -0
  159. digitalocean_genai_sdk/types/knowledge_bases/data_source_create_params.py +33 -0
  160. digitalocean_genai_sdk/types/knowledge_bases/data_source_create_response.py +12 -0
  161. digitalocean_genai_sdk/types/knowledge_bases/data_source_delete_response.py +13 -0
  162. digitalocean_genai_sdk/types/knowledge_bases/data_source_list_params.py +15 -0
  163. digitalocean_genai_sdk/types/knowledge_bases/data_source_list_response.py +18 -0
  164. digitalocean_genai_sdk/types/model.py +21 -0
  165. digitalocean_genai_sdk/types/model_list_response.py +15 -0
  166. digitalocean_genai_sdk/types/providers/__init__.py +3 -0
  167. digitalocean_genai_sdk/types/providers/anthropic/__init__.py +14 -0
  168. digitalocean_genai_sdk/types/providers/anthropic/key_create_params.py +13 -0
  169. digitalocean_genai_sdk/types/providers/anthropic/key_create_response.py +12 -0
  170. digitalocean_genai_sdk/types/providers/anthropic/key_delete_response.py +12 -0
  171. digitalocean_genai_sdk/types/providers/anthropic/key_list_agents_params.py +15 -0
  172. digitalocean_genai_sdk/types/providers/anthropic/key_list_agents_response.py +22 -0
  173. digitalocean_genai_sdk/types/providers/anthropic/key_list_params.py +15 -0
  174. digitalocean_genai_sdk/types/providers/anthropic/key_list_response.py +18 -0
  175. digitalocean_genai_sdk/types/providers/anthropic/key_retrieve_response.py +12 -0
  176. digitalocean_genai_sdk/types/providers/anthropic/key_update_params.py +17 -0
  177. digitalocean_genai_sdk/types/providers/anthropic/key_update_response.py +12 -0
  178. digitalocean_genai_sdk/types/providers/openai/__init__.py +14 -0
  179. digitalocean_genai_sdk/types/providers/openai/key_create_params.py +13 -0
  180. digitalocean_genai_sdk/types/providers/openai/key_create_response.py +12 -0
  181. digitalocean_genai_sdk/types/providers/openai/key_delete_response.py +12 -0
  182. digitalocean_genai_sdk/types/providers/openai/key_list_params.py +15 -0
  183. digitalocean_genai_sdk/types/providers/openai/key_list_response.py +18 -0
  184. digitalocean_genai_sdk/types/providers/openai/key_retrieve_agents_params.py +15 -0
  185. digitalocean_genai_sdk/types/providers/openai/key_retrieve_agents_response.py +22 -0
  186. digitalocean_genai_sdk/types/providers/openai/key_retrieve_response.py +12 -0
  187. digitalocean_genai_sdk/types/providers/openai/key_update_params.py +17 -0
  188. digitalocean_genai_sdk/types/providers/openai/key_update_response.py +12 -0
  189. digitalocean_genai_sdk/types/region_list_params.py +15 -0
  190. digitalocean_genai_sdk/types/region_list_response.py +23 -0
@@ -0,0 +1,151 @@
1
+ from __future__ import annotations
2
+
3
+ import sys
4
+ import typing
5
+ import typing_extensions
6
+ from typing import Any, TypeVar, Iterable, cast
7
+ from collections import abc as _c_abc
8
+ from typing_extensions import (
9
+ TypeIs,
10
+ Required,
11
+ Annotated,
12
+ get_args,
13
+ get_origin,
14
+ )
15
+
16
+ from ._utils import lru_cache
17
+ from .._types import InheritsGeneric
18
+ from .._compat import is_union as _is_union
19
+
20
+
21
+ def is_annotated_type(typ: type) -> bool:
22
+ return get_origin(typ) == Annotated
23
+
24
+
25
+ def is_list_type(typ: type) -> bool:
26
+ return (get_origin(typ) or typ) == list
27
+
28
+
29
+ def is_iterable_type(typ: type) -> bool:
30
+ """If the given type is `typing.Iterable[T]`"""
31
+ origin = get_origin(typ) or typ
32
+ return origin == Iterable or origin == _c_abc.Iterable
33
+
34
+
35
+ def is_union_type(typ: type) -> bool:
36
+ return _is_union(get_origin(typ))
37
+
38
+
39
+ def is_required_type(typ: type) -> bool:
40
+ return get_origin(typ) == Required
41
+
42
+
43
+ def is_typevar(typ: type) -> bool:
44
+ # type ignore is required because type checkers
45
+ # think this expression will always return False
46
+ return type(typ) == TypeVar # type: ignore
47
+
48
+
49
+ _TYPE_ALIAS_TYPES: tuple[type[typing_extensions.TypeAliasType], ...] = (typing_extensions.TypeAliasType,)
50
+ if sys.version_info >= (3, 12):
51
+ _TYPE_ALIAS_TYPES = (*_TYPE_ALIAS_TYPES, typing.TypeAliasType)
52
+
53
+
54
+ def is_type_alias_type(tp: Any, /) -> TypeIs[typing_extensions.TypeAliasType]:
55
+ """Return whether the provided argument is an instance of `TypeAliasType`.
56
+
57
+ ```python
58
+ type Int = int
59
+ is_type_alias_type(Int)
60
+ # > True
61
+ Str = TypeAliasType("Str", str)
62
+ is_type_alias_type(Str)
63
+ # > True
64
+ ```
65
+ """
66
+ return isinstance(tp, _TYPE_ALIAS_TYPES)
67
+
68
+
69
+ # Extracts T from Annotated[T, ...] or from Required[Annotated[T, ...]]
70
+ @lru_cache(maxsize=8096)
71
+ def strip_annotated_type(typ: type) -> type:
72
+ if is_required_type(typ) or is_annotated_type(typ):
73
+ return strip_annotated_type(cast(type, get_args(typ)[0]))
74
+
75
+ return typ
76
+
77
+
78
+ def extract_type_arg(typ: type, index: int) -> type:
79
+ args = get_args(typ)
80
+ try:
81
+ return cast(type, args[index])
82
+ except IndexError as err:
83
+ raise RuntimeError(f"Expected type {typ} to have a type argument at index {index} but it did not") from err
84
+
85
+
86
+ def extract_type_var_from_base(
87
+ typ: type,
88
+ *,
89
+ generic_bases: tuple[type, ...],
90
+ index: int,
91
+ failure_message: str | None = None,
92
+ ) -> type:
93
+ """Given a type like `Foo[T]`, returns the generic type variable `T`.
94
+
95
+ This also handles the case where a concrete subclass is given, e.g.
96
+ ```py
97
+ class MyResponse(Foo[bytes]):
98
+ ...
99
+
100
+ extract_type_var(MyResponse, bases=(Foo,), index=0) -> bytes
101
+ ```
102
+
103
+ And where a generic subclass is given:
104
+ ```py
105
+ _T = TypeVar('_T')
106
+ class MyResponse(Foo[_T]):
107
+ ...
108
+
109
+ extract_type_var(MyResponse[bytes], bases=(Foo,), index=0) -> bytes
110
+ ```
111
+ """
112
+ cls = cast(object, get_origin(typ) or typ)
113
+ if cls in generic_bases: # pyright: ignore[reportUnnecessaryContains]
114
+ # we're given the class directly
115
+ return extract_type_arg(typ, index)
116
+
117
+ # if a subclass is given
118
+ # ---
119
+ # this is needed as __orig_bases__ is not present in the typeshed stubs
120
+ # because it is intended to be for internal use only, however there does
121
+ # not seem to be a way to resolve generic TypeVars for inherited subclasses
122
+ # without using it.
123
+ if isinstance(cls, InheritsGeneric):
124
+ target_base_class: Any | None = None
125
+ for base in cls.__orig_bases__:
126
+ if base.__origin__ in generic_bases:
127
+ target_base_class = base
128
+ break
129
+
130
+ if target_base_class is None:
131
+ raise RuntimeError(
132
+ "Could not find the generic base class;\n"
133
+ "This should never happen;\n"
134
+ f"Does {cls} inherit from one of {generic_bases} ?"
135
+ )
136
+
137
+ extracted = extract_type_arg(target_base_class, index)
138
+ if is_typevar(extracted):
139
+ # If the extracted type argument is itself a type variable
140
+ # then that means the subclass itself is generic, so we have
141
+ # to resolve the type argument from the class itself, not
142
+ # the base class.
143
+ #
144
+ # Note: if there is more than 1 type argument, the subclass could
145
+ # change the ordering of the type arguments, this is not currently
146
+ # supported.
147
+ return extract_type_arg(typ, index)
148
+
149
+ return extracted
150
+
151
+ raise RuntimeError(failure_message or f"Could not resolve inner type variable at index {index} for {typ}")
@@ -0,0 +1,422 @@
1
+ from __future__ import annotations
2
+
3
+ import os
4
+ import re
5
+ import inspect
6
+ import functools
7
+ from typing import (
8
+ Any,
9
+ Tuple,
10
+ Mapping,
11
+ TypeVar,
12
+ Callable,
13
+ Iterable,
14
+ Sequence,
15
+ cast,
16
+ overload,
17
+ )
18
+ from pathlib import Path
19
+ from datetime import date, datetime
20
+ from typing_extensions import TypeGuard
21
+
22
+ import sniffio
23
+
24
+ from .._types import NotGiven, FileTypes, NotGivenOr, HeadersLike
25
+ from .._compat import parse_date as parse_date, parse_datetime as parse_datetime
26
+
27
+ _T = TypeVar("_T")
28
+ _TupleT = TypeVar("_TupleT", bound=Tuple[object, ...])
29
+ _MappingT = TypeVar("_MappingT", bound=Mapping[str, object])
30
+ _SequenceT = TypeVar("_SequenceT", bound=Sequence[object])
31
+ CallableT = TypeVar("CallableT", bound=Callable[..., Any])
32
+
33
+
34
+ def flatten(t: Iterable[Iterable[_T]]) -> list[_T]:
35
+ return [item for sublist in t for item in sublist]
36
+
37
+
38
+ def extract_files(
39
+ # TODO: this needs to take Dict but variance issues.....
40
+ # create protocol type ?
41
+ query: Mapping[str, object],
42
+ *,
43
+ paths: Sequence[Sequence[str]],
44
+ ) -> list[tuple[str, FileTypes]]:
45
+ """Recursively extract files from the given dictionary based on specified paths.
46
+
47
+ A path may look like this ['foo', 'files', '<array>', 'data'].
48
+
49
+ Note: this mutates the given dictionary.
50
+ """
51
+ files: list[tuple[str, FileTypes]] = []
52
+ for path in paths:
53
+ files.extend(_extract_items(query, path, index=0, flattened_key=None))
54
+ return files
55
+
56
+
57
+ def _extract_items(
58
+ obj: object,
59
+ path: Sequence[str],
60
+ *,
61
+ index: int,
62
+ flattened_key: str | None,
63
+ ) -> list[tuple[str, FileTypes]]:
64
+ try:
65
+ key = path[index]
66
+ except IndexError:
67
+ if isinstance(obj, NotGiven):
68
+ # no value was provided - we can safely ignore
69
+ return []
70
+
71
+ # cyclical import
72
+ from .._files import assert_is_file_content
73
+
74
+ # We have exhausted the path, return the entry we found.
75
+ assert flattened_key is not None
76
+
77
+ if is_list(obj):
78
+ files: list[tuple[str, FileTypes]] = []
79
+ for entry in obj:
80
+ assert_is_file_content(entry, key=flattened_key + "[]" if flattened_key else "")
81
+ files.append((flattened_key + "[]", cast(FileTypes, entry)))
82
+ return files
83
+
84
+ assert_is_file_content(obj, key=flattened_key)
85
+ return [(flattened_key, cast(FileTypes, obj))]
86
+
87
+ index += 1
88
+ if is_dict(obj):
89
+ try:
90
+ # We are at the last entry in the path so we must remove the field
91
+ if (len(path)) == index:
92
+ item = obj.pop(key)
93
+ else:
94
+ item = obj[key]
95
+ except KeyError:
96
+ # Key was not present in the dictionary, this is not indicative of an error
97
+ # as the given path may not point to a required field. We also do not want
98
+ # to enforce required fields as the API may differ from the spec in some cases.
99
+ return []
100
+ if flattened_key is None:
101
+ flattened_key = key
102
+ else:
103
+ flattened_key += f"[{key}]"
104
+ return _extract_items(
105
+ item,
106
+ path,
107
+ index=index,
108
+ flattened_key=flattened_key,
109
+ )
110
+ elif is_list(obj):
111
+ if key != "<array>":
112
+ return []
113
+
114
+ return flatten(
115
+ [
116
+ _extract_items(
117
+ item,
118
+ path,
119
+ index=index,
120
+ flattened_key=flattened_key + "[]" if flattened_key is not None else "[]",
121
+ )
122
+ for item in obj
123
+ ]
124
+ )
125
+
126
+ # Something unexpected was passed, just ignore it.
127
+ return []
128
+
129
+
130
+ def is_given(obj: NotGivenOr[_T]) -> TypeGuard[_T]:
131
+ return not isinstance(obj, NotGiven)
132
+
133
+
134
+ # Type safe methods for narrowing types with TypeVars.
135
+ # The default narrowing for isinstance(obj, dict) is dict[unknown, unknown],
136
+ # however this cause Pyright to rightfully report errors. As we know we don't
137
+ # care about the contained types we can safely use `object` in it's place.
138
+ #
139
+ # There are two separate functions defined, `is_*` and `is_*_t` for different use cases.
140
+ # `is_*` is for when you're dealing with an unknown input
141
+ # `is_*_t` is for when you're narrowing a known union type to a specific subset
142
+
143
+
144
+ def is_tuple(obj: object) -> TypeGuard[tuple[object, ...]]:
145
+ return isinstance(obj, tuple)
146
+
147
+
148
+ def is_tuple_t(obj: _TupleT | object) -> TypeGuard[_TupleT]:
149
+ return isinstance(obj, tuple)
150
+
151
+
152
+ def is_sequence(obj: object) -> TypeGuard[Sequence[object]]:
153
+ return isinstance(obj, Sequence)
154
+
155
+
156
+ def is_sequence_t(obj: _SequenceT | object) -> TypeGuard[_SequenceT]:
157
+ return isinstance(obj, Sequence)
158
+
159
+
160
+ def is_mapping(obj: object) -> TypeGuard[Mapping[str, object]]:
161
+ return isinstance(obj, Mapping)
162
+
163
+
164
+ def is_mapping_t(obj: _MappingT | object) -> TypeGuard[_MappingT]:
165
+ return isinstance(obj, Mapping)
166
+
167
+
168
+ def is_dict(obj: object) -> TypeGuard[dict[object, object]]:
169
+ return isinstance(obj, dict)
170
+
171
+
172
+ def is_list(obj: object) -> TypeGuard[list[object]]:
173
+ return isinstance(obj, list)
174
+
175
+
176
+ def is_iterable(obj: object) -> TypeGuard[Iterable[object]]:
177
+ return isinstance(obj, Iterable)
178
+
179
+
180
+ def deepcopy_minimal(item: _T) -> _T:
181
+ """Minimal reimplementation of copy.deepcopy() that will only copy certain object types:
182
+
183
+ - mappings, e.g. `dict`
184
+ - list
185
+
186
+ This is done for performance reasons.
187
+ """
188
+ if is_mapping(item):
189
+ return cast(_T, {k: deepcopy_minimal(v) for k, v in item.items()})
190
+ if is_list(item):
191
+ return cast(_T, [deepcopy_minimal(entry) for entry in item])
192
+ return item
193
+
194
+
195
+ # copied from https://github.com/Rapptz/RoboDanny
196
+ def human_join(seq: Sequence[str], *, delim: str = ", ", final: str = "or") -> str:
197
+ size = len(seq)
198
+ if size == 0:
199
+ return ""
200
+
201
+ if size == 1:
202
+ return seq[0]
203
+
204
+ if size == 2:
205
+ return f"{seq[0]} {final} {seq[1]}"
206
+
207
+ return delim.join(seq[:-1]) + f" {final} {seq[-1]}"
208
+
209
+
210
+ def quote(string: str) -> str:
211
+ """Add single quotation marks around the given string. Does *not* do any escaping."""
212
+ return f"'{string}'"
213
+
214
+
215
+ def required_args(*variants: Sequence[str]) -> Callable[[CallableT], CallableT]:
216
+ """Decorator to enforce a given set of arguments or variants of arguments are passed to the decorated function.
217
+
218
+ Useful for enforcing runtime validation of overloaded functions.
219
+
220
+ Example usage:
221
+ ```py
222
+ @overload
223
+ def foo(*, a: str) -> str: ...
224
+
225
+
226
+ @overload
227
+ def foo(*, b: bool) -> str: ...
228
+
229
+
230
+ # This enforces the same constraints that a static type checker would
231
+ # i.e. that either a or b must be passed to the function
232
+ @required_args(["a"], ["b"])
233
+ def foo(*, a: str | None = None, b: bool | None = None) -> str: ...
234
+ ```
235
+ """
236
+
237
+ def inner(func: CallableT) -> CallableT:
238
+ params = inspect.signature(func).parameters
239
+ positional = [
240
+ name
241
+ for name, param in params.items()
242
+ if param.kind
243
+ in {
244
+ param.POSITIONAL_ONLY,
245
+ param.POSITIONAL_OR_KEYWORD,
246
+ }
247
+ ]
248
+
249
+ @functools.wraps(func)
250
+ def wrapper(*args: object, **kwargs: object) -> object:
251
+ given_params: set[str] = set()
252
+ for i, _ in enumerate(args):
253
+ try:
254
+ given_params.add(positional[i])
255
+ except IndexError:
256
+ raise TypeError(
257
+ f"{func.__name__}() takes {len(positional)} argument(s) but {len(args)} were given"
258
+ ) from None
259
+
260
+ for key in kwargs.keys():
261
+ given_params.add(key)
262
+
263
+ for variant in variants:
264
+ matches = all((param in given_params for param in variant))
265
+ if matches:
266
+ break
267
+ else: # no break
268
+ if len(variants) > 1:
269
+ variations = human_join(
270
+ ["(" + human_join([quote(arg) for arg in variant], final="and") + ")" for variant in variants]
271
+ )
272
+ msg = f"Missing required arguments; Expected either {variations} arguments to be given"
273
+ else:
274
+ assert len(variants) > 0
275
+
276
+ # TODO: this error message is not deterministic
277
+ missing = list(set(variants[0]) - given_params)
278
+ if len(missing) > 1:
279
+ msg = f"Missing required arguments: {human_join([quote(arg) for arg in missing])}"
280
+ else:
281
+ msg = f"Missing required argument: {quote(missing[0])}"
282
+ raise TypeError(msg)
283
+ return func(*args, **kwargs)
284
+
285
+ return wrapper # type: ignore
286
+
287
+ return inner
288
+
289
+
290
+ _K = TypeVar("_K")
291
+ _V = TypeVar("_V")
292
+
293
+
294
+ @overload
295
+ def strip_not_given(obj: None) -> None: ...
296
+
297
+
298
+ @overload
299
+ def strip_not_given(obj: Mapping[_K, _V | NotGiven]) -> dict[_K, _V]: ...
300
+
301
+
302
+ @overload
303
+ def strip_not_given(obj: object) -> object: ...
304
+
305
+
306
+ def strip_not_given(obj: object | None) -> object:
307
+ """Remove all top-level keys where their values are instances of `NotGiven`"""
308
+ if obj is None:
309
+ return None
310
+
311
+ if not is_mapping(obj):
312
+ return obj
313
+
314
+ return {key: value for key, value in obj.items() if not isinstance(value, NotGiven)}
315
+
316
+
317
+ def coerce_integer(val: str) -> int:
318
+ return int(val, base=10)
319
+
320
+
321
+ def coerce_float(val: str) -> float:
322
+ return float(val)
323
+
324
+
325
+ def coerce_boolean(val: str) -> bool:
326
+ return val == "true" or val == "1" or val == "on"
327
+
328
+
329
+ def maybe_coerce_integer(val: str | None) -> int | None:
330
+ if val is None:
331
+ return None
332
+ return coerce_integer(val)
333
+
334
+
335
+ def maybe_coerce_float(val: str | None) -> float | None:
336
+ if val is None:
337
+ return None
338
+ return coerce_float(val)
339
+
340
+
341
+ def maybe_coerce_boolean(val: str | None) -> bool | None:
342
+ if val is None:
343
+ return None
344
+ return coerce_boolean(val)
345
+
346
+
347
+ def removeprefix(string: str, prefix: str) -> str:
348
+ """Remove a prefix from a string.
349
+
350
+ Backport of `str.removeprefix` for Python < 3.9
351
+ """
352
+ if string.startswith(prefix):
353
+ return string[len(prefix) :]
354
+ return string
355
+
356
+
357
+ def removesuffix(string: str, suffix: str) -> str:
358
+ """Remove a suffix from a string.
359
+
360
+ Backport of `str.removesuffix` for Python < 3.9
361
+ """
362
+ if string.endswith(suffix):
363
+ return string[: -len(suffix)]
364
+ return string
365
+
366
+
367
+ def file_from_path(path: str) -> FileTypes:
368
+ contents = Path(path).read_bytes()
369
+ file_name = os.path.basename(path)
370
+ return (file_name, contents)
371
+
372
+
373
+ def get_required_header(headers: HeadersLike, header: str) -> str:
374
+ lower_header = header.lower()
375
+ if is_mapping_t(headers):
376
+ # mypy doesn't understand the type narrowing here
377
+ for k, v in headers.items(): # type: ignore
378
+ if k.lower() == lower_header and isinstance(v, str):
379
+ return v
380
+
381
+ # to deal with the case where the header looks like Stainless-Event-Id
382
+ intercaps_header = re.sub(r"([^\w])(\w)", lambda pat: pat.group(1) + pat.group(2).upper(), header.capitalize())
383
+
384
+ for normalized_header in [header, lower_header, header.upper(), intercaps_header]:
385
+ value = headers.get(normalized_header)
386
+ if value:
387
+ return value
388
+
389
+ raise ValueError(f"Could not find {header} header")
390
+
391
+
392
+ def get_async_library() -> str:
393
+ try:
394
+ return sniffio.current_async_library()
395
+ except Exception:
396
+ return "false"
397
+
398
+
399
+ def lru_cache(*, maxsize: int | None = 128) -> Callable[[CallableT], CallableT]:
400
+ """A version of functools.lru_cache that retains the type signature
401
+ for the wrapped function arguments.
402
+ """
403
+ wrapper = functools.lru_cache( # noqa: TID251
404
+ maxsize=maxsize,
405
+ )
406
+ return cast(Any, wrapper) # type: ignore[no-any-return]
407
+
408
+
409
+ def json_safe(data: object) -> object:
410
+ """Translates a mapping / sequence recursively in the same fashion
411
+ as `pydantic` v2's `model_dump(mode="json")`.
412
+ """
413
+ if is_mapping(data):
414
+ return {json_safe(key): json_safe(value) for key, value in data.items()}
415
+
416
+ if is_iterable(data) and not isinstance(data, (str, bytes, bytearray)):
417
+ return [json_safe(item) for item in data]
418
+
419
+ if isinstance(data, (datetime, date)):
420
+ return data.isoformat()
421
+
422
+ return data
@@ -0,0 +1,4 @@
1
+ # File generated from our OpenAPI spec by Stainless. See CONTRIBUTING.md for details.
2
+
3
+ __title__ = "digitalocean_genai_sdk"
4
+ __version__ = "0.1.0-alpha.3" # x-release-please-version
@@ -0,0 +1,4 @@
1
+ File generated from our OpenAPI spec by Stainless.
2
+
3
+ This directory can be used to store custom files to expand the SDK.
4
+ It is ignored by Stainless code generation and its content (other than this keep file) won't be touched.
File without changes