turbopuffer 0.1.0 → 1.2.0

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 (207) hide show
  1. checksums.yaml +4 -4
  2. data/.ignore +2 -0
  3. data/CHANGELOG.md +205 -0
  4. data/README.md +270 -40
  5. data/SECURITY.md +27 -0
  6. data/lib/turbopuffer/client.rb +122 -16
  7. data/lib/turbopuffer/errors.rb +228 -0
  8. data/lib/turbopuffer/file_part.rb +55 -0
  9. data/lib/turbopuffer/internal/namespace_page.rb +86 -0
  10. data/lib/turbopuffer/internal/transport/base_client.rb +567 -0
  11. data/lib/turbopuffer/internal/transport/pooled_net_requester.rb +201 -0
  12. data/lib/turbopuffer/internal/type/array_of.rb +168 -0
  13. data/lib/turbopuffer/internal/type/base_model.rb +529 -0
  14. data/lib/turbopuffer/internal/type/base_page.rb +55 -0
  15. data/lib/turbopuffer/internal/type/boolean.rb +77 -0
  16. data/lib/turbopuffer/internal/type/converter.rb +327 -0
  17. data/lib/turbopuffer/internal/type/enum.rb +152 -0
  18. data/lib/turbopuffer/internal/type/file_input.rb +108 -0
  19. data/lib/turbopuffer/internal/type/hash_of.rb +188 -0
  20. data/lib/turbopuffer/internal/type/request_parameters.rb +42 -0
  21. data/lib/turbopuffer/internal/type/union.rb +254 -0
  22. data/lib/turbopuffer/internal/type/unknown.rb +81 -0
  23. data/lib/turbopuffer/internal/util.rb +914 -0
  24. data/lib/turbopuffer/internal.rb +20 -0
  25. data/lib/turbopuffer/models/aggregation_group.rb +8 -0
  26. data/lib/turbopuffer/models/attribute_schema.rb +19 -0
  27. data/lib/turbopuffer/models/attribute_schema_config.rb +56 -0
  28. data/lib/turbopuffer/models/attribute_type.rb +7 -0
  29. data/lib/turbopuffer/models/client_namespaces_params.rb +38 -0
  30. data/lib/turbopuffer/models/columns.rb +64 -0
  31. data/lib/turbopuffer/models/distance_metric.rb +19 -0
  32. data/lib/turbopuffer/models/full_text_search.rb +20 -0
  33. data/lib/turbopuffer/models/full_text_search_config.rb +83 -0
  34. data/lib/turbopuffer/models/id.rb +19 -0
  35. data/lib/turbopuffer/models/include_attributes.rb +22 -0
  36. data/lib/turbopuffer/models/language.rb +32 -0
  37. data/lib/turbopuffer/models/namespace_delete_all_params.rb +20 -0
  38. data/lib/turbopuffer/models/namespace_delete_all_response.rb +19 -0
  39. data/lib/turbopuffer/models/namespace_explain_query_params.rb +137 -0
  40. data/lib/turbopuffer/models/namespace_explain_query_response.rb +19 -0
  41. data/lib/turbopuffer/models/namespace_hint_cache_warm_params.rb +20 -0
  42. data/lib/turbopuffer/models/namespace_hint_cache_warm_response.rb +26 -0
  43. data/lib/turbopuffer/models/namespace_metadata.rb +43 -0
  44. data/lib/turbopuffer/models/namespace_metadata_params.rb +20 -0
  45. data/lib/turbopuffer/models/namespace_multi_query_params.rb +73 -0
  46. data/lib/turbopuffer/models/namespace_multi_query_response.rb +58 -0
  47. data/lib/turbopuffer/models/namespace_query_params.rb +137 -0
  48. data/lib/turbopuffer/models/namespace_query_response.rb +49 -0
  49. data/lib/turbopuffer/models/namespace_recall_params.rb +66 -0
  50. data/lib/turbopuffer/models/namespace_recall_response.rb +68 -0
  51. data/lib/turbopuffer/models/namespace_schema_params.rb +20 -0
  52. data/lib/turbopuffer/models/namespace_schema_response.rb +8 -0
  53. data/lib/turbopuffer/models/namespace_summary.rb +19 -0
  54. data/lib/turbopuffer/models/namespace_update_schema_params.rb +29 -0
  55. data/lib/turbopuffer/models/namespace_update_schema_response.rb +9 -0
  56. data/lib/turbopuffer/models/namespace_write_params.rb +158 -0
  57. data/lib/turbopuffer/models/namespace_write_response.rb +67 -0
  58. data/lib/turbopuffer/models/query.rb +81 -0
  59. data/lib/turbopuffer/models/query_billing.rb +26 -0
  60. data/lib/turbopuffer/models/query_performance.rb +63 -0
  61. data/lib/turbopuffer/models/row.rb +46 -0
  62. data/lib/turbopuffer/models/tokenizer.rb +19 -0
  63. data/lib/turbopuffer/models/vector.rb +22 -0
  64. data/lib/turbopuffer/models/vector_encoding.rb +16 -0
  65. data/lib/turbopuffer/models/write_billing.rb +26 -0
  66. data/lib/turbopuffer/models.rb +106 -0
  67. data/lib/turbopuffer/namespace.rb +14 -50
  68. data/lib/turbopuffer/request_options.rb +77 -0
  69. data/lib/turbopuffer/resources/namespaces.rb +366 -0
  70. data/lib/turbopuffer/version.rb +1 -1
  71. data/lib/turbopuffer.rb +92 -4
  72. data/manifest.yaml +15 -0
  73. data/rbi/turbopuffer/client.rbi +91 -0
  74. data/rbi/turbopuffer/errors.rbi +205 -0
  75. data/rbi/turbopuffer/file_part.rbi +37 -0
  76. data/rbi/turbopuffer/internal/namespace_page.rbi +22 -0
  77. data/rbi/turbopuffer/internal/transport/base_client.rbi +297 -0
  78. data/rbi/turbopuffer/internal/transport/pooled_net_requester.rbi +80 -0
  79. data/rbi/turbopuffer/internal/type/array_of.rbi +104 -0
  80. data/rbi/turbopuffer/internal/type/base_model.rbi +304 -0
  81. data/rbi/turbopuffer/internal/type/base_page.rbi +43 -0
  82. data/rbi/turbopuffer/internal/type/boolean.rbi +58 -0
  83. data/rbi/turbopuffer/internal/type/converter.rbi +216 -0
  84. data/rbi/turbopuffer/internal/type/enum.rbi +82 -0
  85. data/rbi/turbopuffer/internal/type/file_input.rbi +59 -0
  86. data/rbi/turbopuffer/internal/type/hash_of.rbi +104 -0
  87. data/rbi/turbopuffer/internal/type/request_parameters.rbi +31 -0
  88. data/rbi/turbopuffer/internal/type/union.rbi +128 -0
  89. data/rbi/turbopuffer/internal/type/unknown.rbi +58 -0
  90. data/rbi/turbopuffer/internal/util.rbi +487 -0
  91. data/rbi/turbopuffer/internal.rbi +18 -0
  92. data/rbi/turbopuffer/models/aggregation_group.rbi +13 -0
  93. data/rbi/turbopuffer/models/attribute_schema.rbi +17 -0
  94. data/rbi/turbopuffer/models/attribute_schema_config.rbi +103 -0
  95. data/rbi/turbopuffer/models/attribute_type.rbi +7 -0
  96. data/rbi/turbopuffer/models/client_namespaces_params.rbi +71 -0
  97. data/rbi/turbopuffer/models/columns.rbi +82 -0
  98. data/rbi/turbopuffer/models/distance_metric.rbi +27 -0
  99. data/rbi/turbopuffer/models/full_text_search.rbi +19 -0
  100. data/rbi/turbopuffer/models/full_text_search_config.rbi +133 -0
  101. data/rbi/turbopuffer/models/id.rbi +16 -0
  102. data/rbi/turbopuffer/models/include_attributes.rbi +24 -0
  103. data/rbi/turbopuffer/models/language.rbi +36 -0
  104. data/rbi/turbopuffer/models/namespace_delete_all_params.rbi +41 -0
  105. data/rbi/turbopuffer/models/namespace_delete_all_response.rbi +31 -0
  106. data/rbi/turbopuffer/models/namespace_explain_query_params.rbi +273 -0
  107. data/rbi/turbopuffer/models/namespace_explain_query_response.rbi +34 -0
  108. data/rbi/turbopuffer/models/namespace_hint_cache_warm_params.rbi +41 -0
  109. data/rbi/turbopuffer/models/namespace_hint_cache_warm_response.rbi +38 -0
  110. data/rbi/turbopuffer/models/namespace_metadata.rbi +62 -0
  111. data/rbi/turbopuffer/models/namespace_metadata_params.rbi +41 -0
  112. data/rbi/turbopuffer/models/namespace_multi_query_params.rbi +175 -0
  113. data/rbi/turbopuffer/models/namespace_multi_query_response.rbi +121 -0
  114. data/rbi/turbopuffer/models/namespace_query_params.rbi +267 -0
  115. data/rbi/turbopuffer/models/namespace_query_response.rbi +84 -0
  116. data/rbi/turbopuffer/models/namespace_recall_params.rbi +106 -0
  117. data/rbi/turbopuffer/models/namespace_recall_response.rbi +134 -0
  118. data/rbi/turbopuffer/models/namespace_schema_params.rbi +41 -0
  119. data/rbi/turbopuffer/models/namespace_schema_response.rbi +11 -0
  120. data/rbi/turbopuffer/models/namespace_summary.rbi +28 -0
  121. data/rbi/turbopuffer/models/namespace_update_schema_params.rbi +80 -0
  122. data/rbi/turbopuffer/models/namespace_update_schema_response.rbi +11 -0
  123. data/rbi/turbopuffer/models/namespace_write_params.rbi +290 -0
  124. data/rbi/turbopuffer/models/namespace_write_response.rbi +101 -0
  125. data/rbi/turbopuffer/models/query.rbi +132 -0
  126. data/rbi/turbopuffer/models/query_billing.rbi +46 -0
  127. data/rbi/turbopuffer/models/query_performance.rbi +82 -0
  128. data/rbi/turbopuffer/models/row.rbi +47 -0
  129. data/rbi/turbopuffer/models/tokenizer.rbi +24 -0
  130. data/rbi/turbopuffer/models/vector.rbi +22 -0
  131. data/rbi/turbopuffer/models/vector_encoding.rbi +22 -0
  132. data/rbi/turbopuffer/models/write_billing.rbi +49 -0
  133. data/rbi/turbopuffer/models.rbi +72 -0
  134. data/rbi/turbopuffer/namespace.rbi +8 -0
  135. data/rbi/turbopuffer/request_options.rbi +59 -0
  136. data/rbi/turbopuffer/resources/namespaces.rbi +304 -0
  137. data/rbi/turbopuffer/version.rbi +5 -0
  138. data/sig/turbopuffer/client.rbs +41 -0
  139. data/sig/turbopuffer/errors.rbs +117 -0
  140. data/sig/turbopuffer/file_part.rbs +21 -0
  141. data/sig/turbopuffer/internal/namespace_page.rbs +13 -0
  142. data/sig/turbopuffer/internal/transport/base_client.rbs +131 -0
  143. data/sig/turbopuffer/internal/transport/pooled_net_requester.rbs +45 -0
  144. data/sig/turbopuffer/internal/type/array_of.rbs +48 -0
  145. data/sig/turbopuffer/internal/type/base_model.rbs +102 -0
  146. data/sig/turbopuffer/internal/type/base_page.rbs +24 -0
  147. data/sig/turbopuffer/internal/type/boolean.rbs +26 -0
  148. data/sig/turbopuffer/internal/type/converter.rbs +79 -0
  149. data/sig/turbopuffer/internal/type/enum.rbs +32 -0
  150. data/sig/turbopuffer/internal/type/file_input.rbs +25 -0
  151. data/sig/turbopuffer/internal/type/hash_of.rbs +48 -0
  152. data/sig/turbopuffer/internal/type/request_parameters.rbs +19 -0
  153. data/sig/turbopuffer/internal/type/union.rbs +52 -0
  154. data/sig/turbopuffer/internal/type/unknown.rbs +26 -0
  155. data/sig/turbopuffer/internal/util.rbs +185 -0
  156. data/sig/turbopuffer/internal.rbs +9 -0
  157. data/sig/turbopuffer/models/aggregation_group.rbs +7 -0
  158. data/sig/turbopuffer/models/attribute_schema.rbs +11 -0
  159. data/sig/turbopuffer/models/attribute_schema_config.rbs +52 -0
  160. data/sig/turbopuffer/models/attribute_type.rbs +5 -0
  161. data/sig/turbopuffer/models/client_namespaces_params.rbs +38 -0
  162. data/sig/turbopuffer/models/columns.rbs +42 -0
  163. data/sig/turbopuffer/models/distance_metric.rbs +17 -0
  164. data/sig/turbopuffer/models/full_text_search.rbs +11 -0
  165. data/sig/turbopuffer/models/full_text_search_config.rbs +75 -0
  166. data/sig/turbopuffer/models/id.rbs +11 -0
  167. data/sig/turbopuffer/models/include_attributes.rbs +13 -0
  168. data/sig/turbopuffer/models/language.rbs +48 -0
  169. data/sig/turbopuffer/models/namespace_delete_all_params.rbs +25 -0
  170. data/sig/turbopuffer/models/namespace_delete_all_response.rbs +13 -0
  171. data/sig/turbopuffer/models/namespace_explain_query_params.rbs +141 -0
  172. data/sig/turbopuffer/models/namespace_explain_query_response.rbs +15 -0
  173. data/sig/turbopuffer/models/namespace_hint_cache_warm_params.rbs +25 -0
  174. data/sig/turbopuffer/models/namespace_hint_cache_warm_response.rbs +18 -0
  175. data/sig/turbopuffer/models/namespace_metadata.rbs +35 -0
  176. data/sig/turbopuffer/models/namespace_metadata_params.rbs +25 -0
  177. data/sig/turbopuffer/models/namespace_multi_query_params.rbs +86 -0
  178. data/sig/turbopuffer/models/namespace_multi_query_response.rbs +65 -0
  179. data/sig/turbopuffer/models/namespace_query_params.rbs +139 -0
  180. data/sig/turbopuffer/models/namespace_query_response.rbs +48 -0
  181. data/sig/turbopuffer/models/namespace_recall_params.rbs +63 -0
  182. data/sig/turbopuffer/models/namespace_recall_response.rbs +61 -0
  183. data/sig/turbopuffer/models/namespace_schema_params.rbs +25 -0
  184. data/sig/turbopuffer/models/namespace_schema_response.rbs +8 -0
  185. data/sig/turbopuffer/models/namespace_summary.rbs +13 -0
  186. data/sig/turbopuffer/models/namespace_update_schema_params.rbs +37 -0
  187. data/sig/turbopuffer/models/namespace_update_schema_response.rbs +8 -0
  188. data/sig/turbopuffer/models/namespace_write_params.rbs +156 -0
  189. data/sig/turbopuffer/models/namespace_write_response.rbs +56 -0
  190. data/sig/turbopuffer/models/query.rbs +75 -0
  191. data/sig/turbopuffer/models/query_billing.rbs +25 -0
  192. data/sig/turbopuffer/models/query_performance.rbs +45 -0
  193. data/sig/turbopuffer/models/row.rbs +24 -0
  194. data/sig/turbopuffer/models/tokenizer.rbs +16 -0
  195. data/sig/turbopuffer/models/vector.rbs +13 -0
  196. data/sig/turbopuffer/models/vector_encoding.rbs +14 -0
  197. data/sig/turbopuffer/models/write_billing.rbs +27 -0
  198. data/sig/turbopuffer/models.rbs +65 -0
  199. data/sig/turbopuffer/namespace.rbs +5 -0
  200. data/sig/turbopuffer/request_options.rbs +36 -0
  201. data/sig/turbopuffer/resources/namespaces.rbs +99 -0
  202. data/sig/turbopuffer/version.rbs +3 -0
  203. metadata +218 -29
  204. data/.standard.yml +0 -3
  205. data/LICENSE.txt +0 -21
  206. data/Rakefile +0 -10
  207. data/sig/turbopuffer.rbs +0 -4
@@ -0,0 +1,567 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Turbopuffer
4
+ module Internal
5
+ module Transport
6
+ # @api private
7
+ #
8
+ # @abstract
9
+ class BaseClient
10
+ extend Turbopuffer::Internal::Util::SorbetRuntimeSupport
11
+
12
+ # from whatwg fetch spec
13
+ MAX_REDIRECTS = 20
14
+
15
+ # rubocop:disable Style/MutableConstant
16
+ PLATFORM_HEADERS =
17
+ {
18
+ "x-stainless-arch" => Turbopuffer::Internal::Util.arch,
19
+ "x-stainless-lang" => "ruby",
20
+ "x-stainless-os" => Turbopuffer::Internal::Util.os,
21
+ "x-stainless-package-version" => Turbopuffer::VERSION,
22
+ "x-stainless-runtime" => ::RUBY_ENGINE,
23
+ "x-stainless-runtime-version" => ::RUBY_ENGINE_VERSION
24
+ }
25
+ # rubocop:enable Style/MutableConstant
26
+
27
+ class << self
28
+ # @api private
29
+ #
30
+ # @param req [Hash{Symbol=>Object}]
31
+ #
32
+ # @raise [ArgumentError]
33
+ def validate!(req)
34
+ keys = [:method, :path, :query, :headers, :body, :unwrap, :page, :stream, :model, :options]
35
+ case req
36
+ in Hash
37
+ req.each_key do |k|
38
+ unless keys.include?(k)
39
+ raise ArgumentError.new("Request `req` keys must be one of #{keys}, got #{k.inspect}")
40
+ end
41
+ end
42
+ else
43
+ raise ArgumentError.new("Request `req` must be a Hash or RequestOptions, got #{req.inspect}")
44
+ end
45
+ end
46
+
47
+ # @api private
48
+ #
49
+ # @param status [Integer]
50
+ # @param headers [Hash{String=>String}]
51
+ #
52
+ # @return [Boolean]
53
+ def should_retry?(status, headers:)
54
+ coerced = Turbopuffer::Internal::Util.coerce_boolean(headers["x-should-retry"])
55
+ case [coerced, status]
56
+ in [true | false, _]
57
+ coerced
58
+ in [_, 408 | 409 | 429 | (500..)]
59
+ # retry on:
60
+ # 408: timeouts
61
+ # 409: locks
62
+ # 429: rate limits
63
+ # 500+: unknown errors
64
+ true
65
+ else
66
+ false
67
+ end
68
+ end
69
+
70
+ # @api private
71
+ #
72
+ # @param request [Hash{Symbol=>Object}] .
73
+ #
74
+ # @option request [Symbol] :method
75
+ #
76
+ # @option request [URI::Generic] :url
77
+ #
78
+ # @option request [Hash{String=>String}] :headers
79
+ #
80
+ # @option request [Object] :body
81
+ #
82
+ # @option request [Integer] :max_retries
83
+ #
84
+ # @option request [Float] :timeout
85
+ #
86
+ # @param status [Integer]
87
+ #
88
+ # @param response_headers [Hash{String=>String}]
89
+ #
90
+ # @return [Hash{Symbol=>Object}]
91
+ def follow_redirect(request, status:, response_headers:)
92
+ method, url, headers = request.fetch_values(:method, :url, :headers)
93
+ location =
94
+ Kernel.then do
95
+ URI.join(url, response_headers["location"])
96
+ rescue ArgumentError
97
+ message = "Server responded with status #{status} but no valid location header."
98
+ raise Turbopuffer::Errors::APIConnectionError.new(
99
+ url: url,
100
+ response: response_headers,
101
+ message: message
102
+ )
103
+ end
104
+
105
+ request = {**request, url: location}
106
+
107
+ case [url.scheme, location.scheme]
108
+ in ["https", "http"]
109
+ message = "Tried to redirect to a insecure URL"
110
+ raise Turbopuffer::Errors::APIConnectionError.new(
111
+ url: url,
112
+ response: response_headers,
113
+ message: message
114
+ )
115
+ else
116
+ nil
117
+ end
118
+
119
+ # from whatwg fetch spec
120
+ case [status, method]
121
+ in [301 | 302, :post] | [303, _]
122
+ drop = %w[content-encoding content-language content-length content-location content-type]
123
+ request = {
124
+ **request,
125
+ method: method == :head ? :head : :get,
126
+ headers: headers.except(*drop),
127
+ body: nil
128
+ }
129
+ else
130
+ end
131
+
132
+ # from undici
133
+ if Turbopuffer::Internal::Util.uri_origin(url) != Turbopuffer::Internal::Util.uri_origin(location)
134
+ drop = %w[authorization cookie host proxy-authorization]
135
+ request = {**request, headers: request.fetch(:headers).except(*drop)}
136
+ end
137
+
138
+ request
139
+ end
140
+
141
+ # @api private
142
+ #
143
+ # @param status [Integer, Turbopuffer::Errors::APIConnectionError]
144
+ # @param stream [Enumerable<String>, nil]
145
+ def reap_connection!(status, stream:)
146
+ case status
147
+ in (..199) | (300..499)
148
+ stream&.each { next }
149
+ in Turbopuffer::Errors::APIConnectionError | (500..)
150
+ Turbopuffer::Internal::Util.close_fused!(stream)
151
+ else
152
+ end
153
+ end
154
+ end
155
+
156
+ # @return [URI::Generic]
157
+ attr_reader :base_url
158
+
159
+ # @return [Float]
160
+ attr_reader :timeout
161
+
162
+ # @return [Integer]
163
+ attr_reader :max_retries
164
+
165
+ # @return [Float]
166
+ attr_reader :initial_retry_delay
167
+
168
+ # @return [Float]
169
+ attr_reader :max_retry_delay
170
+
171
+ # @return [Hash{String=>String}]
172
+ attr_reader :headers
173
+
174
+ # @return [String, nil]
175
+ attr_reader :idempotency_header
176
+
177
+ # @api private
178
+ # @return [Turbopuffer::Internal::Transport::PooledNetRequester]
179
+ attr_reader :requester
180
+
181
+ # @api private
182
+ #
183
+ # @param base_url [String]
184
+ # @param timeout [Float]
185
+ # @param max_retries [Integer]
186
+ # @param initial_retry_delay [Float]
187
+ # @param max_retry_delay [Float]
188
+ # @param headers [Hash{String=>String, Integer, Array<String, Integer, nil>, nil}]
189
+ # @param idempotency_header [String, nil]
190
+ def initialize(
191
+ base_url:,
192
+ timeout: 0.0,
193
+ max_retries: 0,
194
+ initial_retry_delay: 0.0,
195
+ max_retry_delay: 0.0,
196
+ headers: {},
197
+ idempotency_header: nil
198
+ )
199
+ @requester = Turbopuffer::Internal::Transport::PooledNetRequester.new
200
+ @headers = Turbopuffer::Internal::Util.normalized_headers(
201
+ self.class::PLATFORM_HEADERS,
202
+ {
203
+ "accept" => "application/json",
204
+ "content-type" => "application/json"
205
+ },
206
+ headers
207
+ )
208
+ @base_url_components = Turbopuffer::Internal::Util.parse_uri(base_url)
209
+ @base_url = Turbopuffer::Internal::Util.unparse_uri(@base_url_components)
210
+ @idempotency_header = idempotency_header&.to_s&.downcase
211
+ @timeout = timeout
212
+ @max_retries = max_retries
213
+ @initial_retry_delay = initial_retry_delay
214
+ @max_retry_delay = max_retry_delay
215
+ end
216
+
217
+ # @api private
218
+ #
219
+ # @return [Hash{String=>String}]
220
+ private def auth_headers = {}
221
+
222
+ # @api private
223
+ #
224
+ # @return [String]
225
+ private def generate_idempotency_key = "stainless-ruby-retry-#{SecureRandom.uuid}"
226
+
227
+ # @api private
228
+ #
229
+ # @param req [Hash{Symbol=>Object}] .
230
+ #
231
+ # @option req [Symbol] :method
232
+ #
233
+ # @option req [String, Array<String>] :path
234
+ #
235
+ # @option req [Hash{String=>Array<String>, String, nil}, nil] :query
236
+ #
237
+ # @option req [Hash{String=>String, Integer, Array<String, Integer, nil>, nil}, nil] :headers
238
+ #
239
+ # @option req [Object, nil] :body
240
+ #
241
+ # @option req [Symbol, Integer, Array<Symbol, Integer>, Proc, nil] :unwrap
242
+ #
243
+ # @option req [Class<Turbopuffer::Internal::Type::BasePage>, nil] :page
244
+ #
245
+ # @option req [Class<Turbopuffer::Internal::Type::BaseStream>, nil] :stream
246
+ #
247
+ # @option req [Turbopuffer::Internal::Type::Converter, Class, nil] :model
248
+ #
249
+ # @param opts [Hash{Symbol=>Object}] .
250
+ #
251
+ # @option opts [String, nil] :idempotency_key
252
+ #
253
+ # @option opts [Hash{String=>Array<String>, String, nil}, nil] :extra_query
254
+ #
255
+ # @option opts [Hash{String=>String, nil}, nil] :extra_headers
256
+ #
257
+ # @option opts [Object, nil] :extra_body
258
+ #
259
+ # @option opts [Integer, nil] :max_retries
260
+ #
261
+ # @option opts [Float, nil] :timeout
262
+ #
263
+ # @return [Hash{Symbol=>Object}]
264
+ private def build_request(req, opts)
265
+ method, uninterpolated_path = req.fetch_values(:method, :path)
266
+
267
+ path = Turbopuffer::Internal::Util.interpolate_path(uninterpolated_path)
268
+
269
+ query = Turbopuffer::Internal::Util.deep_merge(req[:query].to_h, opts[:extra_query].to_h)
270
+
271
+ headers = Turbopuffer::Internal::Util.normalized_headers(
272
+ @headers,
273
+ auth_headers,
274
+ req[:headers].to_h,
275
+ opts[:extra_headers].to_h
276
+ )
277
+
278
+ if @idempotency_header &&
279
+ !headers.key?(@idempotency_header) &&
280
+ (!Net::HTTP::IDEMPOTENT_METHODS_.include?(method.to_s.upcase) || opts.key?(:idempotency_key))
281
+ headers[@idempotency_header] = opts.fetch(:idempotency_key) { generate_idempotency_key }
282
+ end
283
+
284
+ unless headers.key?("x-stainless-retry-count")
285
+ headers["x-stainless-retry-count"] = "0"
286
+ end
287
+
288
+ timeout = opts.fetch(:timeout, @timeout).to_f.clamp(0..)
289
+ unless headers.key?("x-stainless-timeout") || timeout.zero?
290
+ headers["x-stainless-timeout"] = timeout.to_s
291
+ end
292
+
293
+ headers.reject! { |_, v| v.to_s.empty? }
294
+
295
+ body =
296
+ case method
297
+ in :get | :head | :options | :trace
298
+ nil
299
+ else
300
+ Turbopuffer::Internal::Util.deep_merge(*[req[:body], opts[:extra_body]].compact)
301
+ end
302
+
303
+ url = Turbopuffer::Internal::Util.join_parsed_uri(
304
+ @base_url_components,
305
+ {**req, path: path, query: query}
306
+ )
307
+ headers, encoded = Turbopuffer::Internal::Util.encode_content(headers, body)
308
+ {
309
+ method: method,
310
+ url: url,
311
+ headers: headers,
312
+ body: encoded,
313
+ max_retries: opts.fetch(:max_retries, @max_retries),
314
+ timeout: timeout
315
+ }
316
+ end
317
+
318
+ # @api private
319
+ #
320
+ # @param headers [Hash{String=>String}]
321
+ # @param retry_count [Integer]
322
+ #
323
+ # @return [Float]
324
+ private def retry_delay(headers, retry_count:)
325
+ # Non-standard extension
326
+ span = Float(headers["retry-after-ms"], exception: false)&.then { _1 / 1000 }
327
+ return span if span
328
+
329
+ retry_header = headers["retry-after"]
330
+ return span if (span = Float(retry_header, exception: false))
331
+
332
+ span = retry_header&.then do
333
+ Time.httpdate(_1) - Time.now
334
+ rescue ArgumentError
335
+ nil
336
+ end
337
+ return span if span
338
+
339
+ scale = retry_count**2
340
+ jitter = 1 - (0.25 * rand)
341
+ (@initial_retry_delay * scale * jitter).clamp(0, @max_retry_delay)
342
+ end
343
+
344
+ # @api private
345
+ #
346
+ # @param request [Hash{Symbol=>Object}] .
347
+ #
348
+ # @option request [Symbol] :method
349
+ #
350
+ # @option request [URI::Generic] :url
351
+ #
352
+ # @option request [Hash{String=>String}] :headers
353
+ #
354
+ # @option request [Object] :body
355
+ #
356
+ # @option request [Integer] :max_retries
357
+ #
358
+ # @option request [Float] :timeout
359
+ #
360
+ # @param redirect_count [Integer]
361
+ #
362
+ # @param retry_count [Integer]
363
+ #
364
+ # @param send_retry_header [Boolean]
365
+ #
366
+ # @raise [Turbopuffer::Errors::APIError]
367
+ # @return [Array(Integer, Net::HTTPResponse, Enumerable<String>)]
368
+ def send_request(request, redirect_count:, retry_count:, send_retry_header:)
369
+ url, headers, max_retries, timeout = request.fetch_values(:url, :headers, :max_retries, :timeout)
370
+ input = {**request.except(:timeout), deadline: Turbopuffer::Internal::Util.monotonic_secs + timeout}
371
+
372
+ if send_retry_header
373
+ headers["x-stainless-retry-count"] = retry_count.to_s
374
+ end
375
+
376
+ begin
377
+ status, response, stream = @requester.execute(input)
378
+ rescue Turbopuffer::Errors::APIConnectionError => e
379
+ status = e
380
+ end
381
+ headers = Turbopuffer::Internal::Util.normalized_headers(response&.each_header&.to_h)
382
+
383
+ case status
384
+ in ..299
385
+ [status, response, stream]
386
+ in 300..399 if redirect_count >= self.class::MAX_REDIRECTS
387
+ self.class.reap_connection!(status, stream: stream)
388
+
389
+ message = "Failed to complete the request within #{self.class::MAX_REDIRECTS} redirects."
390
+ raise Turbopuffer::Errors::APIConnectionError.new(url: url, response: response, message: message)
391
+ in 300..399
392
+ self.class.reap_connection!(status, stream: stream)
393
+
394
+ request = self.class.follow_redirect(request, status: status, response_headers: headers)
395
+ send_request(
396
+ request,
397
+ redirect_count: redirect_count + 1,
398
+ retry_count: retry_count,
399
+ send_retry_header: send_retry_header
400
+ )
401
+ in Turbopuffer::Errors::APIConnectionError if retry_count >= max_retries
402
+ raise status
403
+ in (400..) if retry_count >= max_retries || !self.class.should_retry?(status, headers: headers)
404
+ decoded = Kernel.then do
405
+ Turbopuffer::Internal::Util.decode_content(headers, stream: stream, suppress_error: true)
406
+ ensure
407
+ self.class.reap_connection!(status, stream: stream)
408
+ end
409
+
410
+ raise Turbopuffer::Errors::APIStatusError.for(
411
+ url: url,
412
+ status: status,
413
+ headers: headers,
414
+ body: decoded,
415
+ request: nil,
416
+ response: response
417
+ )
418
+ in (400..) | Turbopuffer::Errors::APIConnectionError
419
+ self.class.reap_connection!(status, stream: stream)
420
+
421
+ delay = retry_delay(response || {}, retry_count: retry_count)
422
+ sleep(delay)
423
+
424
+ send_request(
425
+ request,
426
+ redirect_count: redirect_count,
427
+ retry_count: retry_count + 1,
428
+ send_retry_header: send_retry_header
429
+ )
430
+ end
431
+ end
432
+
433
+ # Execute the request specified by `req`. This is the method that all resource
434
+ # methods call into.
435
+ #
436
+ # @overload request(method, path, query: {}, headers: {}, body: nil, unwrap: nil, page: nil, stream: nil, model: Turbopuffer::Internal::Type::Unknown, options: {})
437
+ #
438
+ # @param method [Symbol]
439
+ #
440
+ # @param path [String, Array<String>]
441
+ #
442
+ # @param query [Hash{String=>Array<String>, String, nil}, nil]
443
+ #
444
+ # @param headers [Hash{String=>String, Integer, Array<String, Integer, nil>, nil}, nil]
445
+ #
446
+ # @param body [Object, nil]
447
+ #
448
+ # @param unwrap [Symbol, Integer, Array<Symbol, Integer>, Proc, nil]
449
+ #
450
+ # @param page [Class<Turbopuffer::Internal::Type::BasePage>, nil]
451
+ #
452
+ # @param stream [Class<Turbopuffer::Internal::Type::BaseStream>, nil]
453
+ #
454
+ # @param model [Turbopuffer::Internal::Type::Converter, Class, nil]
455
+ #
456
+ # @param options [Turbopuffer::RequestOptions, Hash{Symbol=>Object}, nil] .
457
+ #
458
+ # @option options [String, nil] :idempotency_key
459
+ #
460
+ # @option options [Hash{String=>Array<String>, String, nil}, nil] :extra_query
461
+ #
462
+ # @option options [Hash{String=>String, nil}, nil] :extra_headers
463
+ #
464
+ # @option options [Object, nil] :extra_body
465
+ #
466
+ # @option options [Integer, nil] :max_retries
467
+ #
468
+ # @option options [Float, nil] :timeout
469
+ #
470
+ # @raise [Turbopuffer::Errors::APIError]
471
+ # @return [Object]
472
+ def request(req)
473
+ self.class.validate!(req)
474
+ model = req.fetch(:model) { Turbopuffer::Internal::Type::Unknown }
475
+ opts = req[:options].to_h
476
+ unwrap = req[:unwrap]
477
+ Turbopuffer::RequestOptions.validate!(opts)
478
+ request = build_request(req.except(:options), opts)
479
+ url = request.fetch(:url)
480
+
481
+ # Don't send the current retry count in the headers if the caller modified the header defaults.
482
+ send_retry_header = request.fetch(:headers)["x-stainless-retry-count"] == "0"
483
+ status, response, stream = send_request(
484
+ request,
485
+ redirect_count: 0,
486
+ retry_count: 0,
487
+ send_retry_header: send_retry_header
488
+ )
489
+
490
+ headers = Turbopuffer::Internal::Util.normalized_headers(response.each_header.to_h)
491
+ decoded = Turbopuffer::Internal::Util.decode_content(headers, stream: stream)
492
+ case req
493
+ in {stream: Class => st}
494
+ st.new(
495
+ model: model,
496
+ url: url,
497
+ status: status,
498
+ headers: headers,
499
+ response: response,
500
+ unwrap: unwrap,
501
+ stream: decoded
502
+ )
503
+ in {page: Class => page}
504
+ page.new(client: self, req: req, headers: headers, page_data: decoded)
505
+ else
506
+ unwrapped = Turbopuffer::Internal::Util.dig(decoded, unwrap)
507
+ Turbopuffer::Internal::Type::Converter.coerce(model, unwrapped)
508
+ end
509
+ end
510
+
511
+ # @api private
512
+ #
513
+ # @return [String]
514
+ def inspect
515
+ # rubocop:disable Layout/LineLength
516
+ "#<#{self.class.name}:0x#{object_id.to_s(16)} base_url=#{@base_url} max_retries=#{@max_retries} timeout=#{@timeout}>"
517
+ # rubocop:enable Layout/LineLength
518
+ end
519
+
520
+ define_sorbet_constant!(:RequestComponents) do
521
+ T.type_alias do
522
+ {
523
+ method: Symbol,
524
+ path: T.any(String, T::Array[String]),
525
+ query: T.nilable(T::Hash[String, T.nilable(T.any(T::Array[String], String))]),
526
+ headers: T.nilable(
527
+ T::Hash[String,
528
+ T.nilable(
529
+ T.any(
530
+ String,
531
+ Integer,
532
+ T::Array[T.nilable(T.any(String, Integer))]
533
+ )
534
+ )]
535
+ ),
536
+ body: T.nilable(T.anything),
537
+ unwrap: T.nilable(
538
+ T.any(
539
+ Symbol,
540
+ Integer,
541
+ T::Array[T.any(Symbol, Integer)],
542
+ T.proc.params(arg0: T.anything).returns(T.anything)
543
+ )
544
+ ),
545
+ page: T.nilable(T::Class[Turbopuffer::Internal::Type::BasePage[Turbopuffer::Internal::Type::BaseModel]]),
546
+ stream: T.nilable(T::Class[T.anything]),
547
+ model: T.nilable(Turbopuffer::Internal::Type::Converter::Input),
548
+ options: T.nilable(Turbopuffer::RequestOptions::OrHash)
549
+ }
550
+ end
551
+ end
552
+ define_sorbet_constant!(:RequestInput) do
553
+ T.type_alias do
554
+ {
555
+ method: Symbol,
556
+ url: URI::Generic,
557
+ headers: T::Hash[String, String],
558
+ body: T.anything,
559
+ max_retries: Integer,
560
+ timeout: Float
561
+ }
562
+ end
563
+ end
564
+ end
565
+ end
566
+ end
567
+ end