toolcraft_request/
client.rs1use futures_util::StreamExt;
2use reqwest::{Client, multipart};
3use url::Url;
4
5use crate::{
6 error::{Error, Result},
7 header_map::HeaderMap,
8 response::{ByteStream, Response},
9};
10
11#[derive(Debug)]
13pub struct Request {
14 client: Client,
15 base_url: Option<Url>,
16 default_headers: HeaderMap,
17}
18
19impl Request {
20 pub fn new() -> Result<Self> {
22 let client = Client::builder()
23 .build()
24 .map_err(|e| Error::ErrorMessage(e.to_string().into()))?;
25 Ok(Request {
26 client,
27 base_url: None,
28 default_headers: HeaderMap::new(),
29 })
30 }
31
32 pub fn with_timeout(timeout_sec: u64) -> Result<Self> {
33 let client = Client::builder()
34 .timeout(std::time::Duration::from_secs(timeout_sec))
35 .build()
36 .map_err(|e| Error::ErrorMessage(e.to_string().into()))?;
37 Ok(Request {
38 client,
39 base_url: None,
40 default_headers: HeaderMap::new(),
41 })
42 }
43
44 pub fn set_base_url(&mut self, base_url: &str) -> Result<()> {
46 let mut url_str = base_url.to_string();
47 if !url_str.ends_with('/') {
48 url_str.push('/');
49 }
50 let url = Url::parse(&url_str)?;
51 self.base_url = Some(url);
52 Ok(())
53 }
54
55 pub fn set_default_headers(&mut self, headers: HeaderMap) {
57 self.default_headers = headers;
58 }
59
60 pub async fn get(
62 &self,
63 endpoint: &str,
64 query: Option<Vec<(String, String)>>,
65 headers: Option<HeaderMap>,
66 ) -> Result<Response> {
67 let url = self.build_url(endpoint, query)?;
68 let mut request = self.client.get(url.as_str());
69
70 let mut combined_headers = self.default_headers.clone();
71 if let Some(custom_headers) = headers {
72 combined_headers.merge(custom_headers);
73 }
74 request = request.headers(combined_headers.inner().clone());
75
76 let response = request.send().await?;
77 Ok(response.into())
78 }
79
80 pub async fn post(
82 &self,
83 endpoint: &str,
84 body: &serde_json::Value,
85 headers: Option<HeaderMap>,
86 ) -> Result<Response> {
87 let url = self.build_url(endpoint, None)?;
88 let mut request = self.client.post(url).json(body);
89
90 let mut combined_headers = self.default_headers.clone();
91 if let Some(custom_headers) = headers {
92 combined_headers.merge(custom_headers);
93 }
94 request = request.headers(combined_headers.inner().clone());
95
96 let response = request.send().await?;
97 Ok(response.into())
98 }
99
100 pub async fn put(
102 &self,
103 endpoint: &str,
104 body: &serde_json::Value,
105 headers: Option<HeaderMap>,
106 ) -> Result<Response> {
107 let url = self.build_url(endpoint, None)?;
108 let mut request = self.client.put(url).json(body);
109
110 let mut combined_headers = self.default_headers.clone();
111 if let Some(custom_headers) = headers {
112 combined_headers.merge(custom_headers);
113 }
114 request = request.headers(combined_headers.inner().clone());
115
116 let response = request.send().await?;
117 Ok(response.into())
118 }
119
120 pub async fn delete(
122 &self,
123 endpoint: &str,
124 headers: Option<HeaderMap>,
125 ) -> Result<Response> {
126 let url = self.build_url(endpoint, None)?;
127 let mut request = self.client.delete(url);
128
129 let mut combined_headers = self.default_headers.clone();
130 if let Some(custom_headers) = headers {
131 combined_headers.merge(custom_headers);
132 }
133 request = request.headers(combined_headers.inner().clone());
134
135 let response = request.send().await?;
136 Ok(response.into())
137 }
138
139 pub async fn post_form(
165 &self,
166 endpoint: &str,
167 form_fields: Vec<FormField>,
168 headers: Option<HeaderMap>,
169 ) -> Result<Response> {
170 let url = self.build_url(endpoint, None)?;
171
172 let mut form = multipart::Form::new();
173 for field in form_fields {
174 match field {
175 FormField::Text { name, value } => {
176 form = form.text(name, value);
177 }
178 FormField::File {
179 name,
180 filename,
181 content,
182 } => {
183 let part = multipart::Part::bytes(content).file_name(filename);
184 form = form.part(name, part);
185 }
186 }
187 }
188
189 let mut combined_headers = self.default_headers.clone();
190 if let Some(custom_headers) = headers {
191 combined_headers.merge(custom_headers);
192 }
193
194 combined_headers.remove("Content-Type");
196 combined_headers.remove("content-type");
197
198 let mut request = self.client.post(url).multipart(form);
199 request = request.headers(combined_headers.inner().clone());
200
201 let response = request.send().await?;
202 Ok(response.into())
203 }
204
205 pub async fn post_stream(
207 &self,
208 endpoint: &str,
209 body: &serde_json::Value,
210 headers: Option<HeaderMap>,
211 ) -> Result<ByteStream> {
212 let url = self.build_url(endpoint, None)?;
213 let mut request = self.client.post(url).json(body);
214
215 let mut combined_headers = self.default_headers.clone();
216 if let Some(custom_headers) = headers {
217 combined_headers.merge(custom_headers);
218 }
219 request = request.headers(combined_headers.inner().clone());
220
221 let response = request.send().await?;
222 if !response.status().is_success() {
223 return Err(Error::ErrorMessage(
224 format!("Unexpected status: {}", response.status()).into(),
225 ));
226 }
227
228 let stream = response
229 .bytes_stream()
230 .map(|chunk_result| chunk_result.map_err(Error::from));
231 Ok(Box::pin(stream))
232 }
233
234 fn build_url(&self, endpoint: &str, query: Option<Vec<(String, String)>>) -> Result<Url> {
236 let mut url = if let Some(base_url) = &self.base_url {
237 base_url.join(endpoint)?
238 } else {
239 Url::parse(endpoint)?
240 };
241
242 if let Some(query_params) = query {
243 let query_pairs: Vec<(String, String)> = query_params.into_iter().collect();
244 url.query_pairs_mut().extend_pairs(query_pairs);
245 }
246
247 Ok(url)
248 }
249}
250
251pub fn parse_url(url: &str, query: Option<Vec<(String, String)>>) -> Result<Url> {
253 let mut url = Url::parse(url)?;
254 if let Some(query_params) = query {
255 let query_pairs: Vec<(String, String)> = query_params.into_iter().collect();
256 url.query_pairs_mut().extend_pairs(query_pairs);
257 }
258 Ok(url)
259}
260
261#[derive(Debug, Clone)]
263pub enum FormField {
264 Text { name: String, value: String },
266 File {
268 name: String,
269 filename: String,
270 content: Vec<u8>,
271 },
272}
273
274impl FormField {
275 pub fn text(name: impl Into<String>, value: impl Into<String>) -> Self {
283 FormField::Text {
284 name: name.into(),
285 value: value.into(),
286 }
287 }
288
289 pub fn file_from_bytes(
298 name: impl Into<String>,
299 filename: impl Into<String>,
300 content: Vec<u8>,
301 ) -> Self {
302 FormField::File {
303 name: name.into(),
304 filename: filename.into(),
305 content,
306 }
307 }
308
309 pub async fn file(name: impl Into<String>, path: impl AsRef<std::path::Path>) -> Result<Self> {
321 let path = path.as_ref();
322 let filename = path
323 .file_name()
324 .and_then(|n| n.to_str())
325 .ok_or_else(|| Error::ErrorMessage("Invalid file path".into()))?
326 .to_string();
327
328 let content = tokio::fs::read(path)
329 .await
330 .map_err(|e| Error::ErrorMessage(format!("Failed to read file: {}", e).into()))?;
331
332 Ok(FormField::File {
333 name: name.into(),
334 filename,
335 content,
336 })
337 }
338}