toolcraft_request/
client.rs

1use 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/// An HTTP request builder and executor with base URL and default headers.
12#[derive(Debug)]
13pub struct Request {
14    client: Client,
15    base_url: Option<Url>,
16    default_headers: HeaderMap,
17}
18
19impl Request {
20    /// Create a new Request client.
21    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    /// Set the base URL for all requests.
45    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    /// Set default headers to be applied on all requests.
56    pub fn set_default_headers(&mut self, headers: HeaderMap) {
57        self.default_headers = headers;
58    }
59
60    /// Send a GET request.
61    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    /// Send a POST request with JSON body.
81    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    /// Send a PUT request with JSON body.
101    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    /// Send a DELETE request.
121    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    /// Send a POST request with multipart/form-data.
140    ///
141    /// # Arguments
142    /// * `endpoint` - The URL endpoint
143    /// * `form_fields` - Vector of form fields (text or file)
144    /// * `headers` - Optional custom headers
145    ///
146    /// # Important
147    /// The `Content-Type` header will be automatically removed from default and custom headers
148    /// to allow reqwest to set the correct `multipart/form-data` with boundary.
149    ///
150    /// # Example
151    /// ```no_run
152    /// use toolcraft_request::{FormField, Request};
153    ///
154    /// # async fn example() -> Result<(), Box<dyn std::error::Error>> {
155    /// let client = Request::new()?;
156    /// let fields = vec![
157    ///     FormField::text("name", "John"),
158    ///     FormField::file("avatar", "/path/to/image.jpg").await?,
159    /// ];
160    /// let response = client.post_form("/upload", fields, None).await?;
161    /// # Ok(())
162    /// # }
163    /// ```
164    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        // Remove Content-Type to let reqwest set the correct multipart/form-data with boundary
195        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    /// Send a streaming POST request and return the response stream.
206    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    /// Build a full URL by combining base URL, endpoint, and optional query parameters.
235    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
251/// Parse a full URL with optional query parameters.
252pub 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/// Represents a field in a multipart/form-data request.
262#[derive(Debug, Clone)]
263pub enum FormField {
264    /// A text field.
265    Text { name: String, value: String },
266    /// A file field.
267    File {
268        name: String,
269        filename: String,
270        content: Vec<u8>,
271    },
272}
273
274impl FormField {
275    /// Create a text field.
276    ///
277    /// # Example
278    /// ```
279    /// use toolcraft_request::FormField;
280    /// let field = FormField::text("username", "john_doe");
281    /// ```
282    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    /// Create a file field from bytes.
290    ///
291    /// # Example
292    /// ```
293    /// use toolcraft_request::FormField;
294    /// let data = b"file content".to_vec();
295    /// let field = FormField::file_from_bytes("avatar", "photo.jpg", data);
296    /// ```
297    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    /// Create a file field by reading from a file path.
310    ///
311    /// # Example
312    /// ```no_run
313    /// use toolcraft_request::FormField;
314    ///
315    /// # async fn example() -> Result<(), Box<dyn std::error::Error>> {
316    /// let field = FormField::file("avatar", "/path/to/image.jpg").await?;
317    /// # Ok(())
318    /// # }
319    /// ```
320    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}