openai_api_rs/v1/
api.rs

1use crate::v1::assistant::{
2    AssistantFileObject, AssistantFileRequest, AssistantObject, AssistantRequest, ListAssistant,
3    ListAssistantFile,
4};
5use crate::v1::audio::{
6    AudioSpeechRequest, AudioSpeechResponse, AudioTranscriptionRequest, AudioTranscriptionResponse,
7    AudioTranslationRequest, AudioTranslationResponse,
8};
9use crate::v1::batch::{BatchResponse, CreateBatchRequest, ListBatchResponse};
10use crate::v1::chat_completion::{ChatCompletionRequest, ChatCompletionResponse};
11use crate::v1::common;
12use crate::v1::completion::{CompletionRequest, CompletionResponse};
13use crate::v1::edit::{EditRequest, EditResponse};
14use crate::v1::embedding::{EmbeddingRequest, EmbeddingResponse};
15use crate::v1::error::APIError;
16use crate::v1::file::{
17    FileDeleteRequest, FileDeleteResponse, FileListResponse, FileRetrieveResponse,
18    FileUploadRequest, FileUploadResponse,
19};
20use crate::v1::fine_tuning::{
21    CancelFineTuningJobRequest, CreateFineTuningJobRequest, FineTuningJobEvent,
22    FineTuningJobObject, FineTuningPagination, ListFineTuningJobEventsRequest,
23    RetrieveFineTuningJobRequest,
24};
25use crate::v1::image::{
26    ImageEditRequest, ImageEditResponse, ImageGenerationRequest, ImageGenerationResponse,
27    ImageVariationRequest, ImageVariationResponse,
28};
29use crate::v1::message::{
30    CreateMessageRequest, ListMessage, ListMessageFile, MessageFileObject, MessageObject,
31    ModifyMessageRequest,
32};
33use crate::v1::model::{ModelResponse, ModelsResponse};
34use crate::v1::moderation::{CreateModerationRequest, CreateModerationResponse};
35use crate::v1::run::{
36    CreateRunRequest, CreateThreadAndRunRequest, ListRun, ListRunStep, ModifyRunRequest, RunObject,
37    RunStepObject,
38};
39use crate::v1::thread::{CreateThreadRequest, ModifyThreadRequest, ThreadObject};
40
41use bytes::Bytes;
42use reqwest::header::{HeaderMap, HeaderName, HeaderValue};
43use reqwest::multipart::{Form, Part};
44use reqwest::{Client, Method, Response};
45use serde::Serialize;
46use serde_json::Value;
47use url::Url;
48
49use std::error::Error;
50use std::fs::{create_dir_all, File};
51use std::io::Read;
52use std::io::Write;
53use std::path::Path;
54
55const API_URL_V1: &str = "https://api.openai.com/v1";
56
57#[derive(Default)]
58pub struct OpenAIClientBuilder {
59    api_endpoint: Option<String>,
60    api_key: Option<String>,
61    organization: Option<String>,
62    proxy: Option<String>,
63    timeout: Option<u64>,
64    headers: Option<HeaderMap>,
65}
66
67#[derive(Debug)]
68pub struct OpenAIClient {
69    api_endpoint: String,
70    api_key: Option<String>,
71    organization: Option<String>,
72    proxy: Option<String>,
73    timeout: Option<u64>,
74    headers: Option<HeaderMap>,
75    pub response_headers: Option<HeaderMap>,
76}
77
78impl OpenAIClientBuilder {
79    pub fn new() -> Self {
80        Self::default()
81    }
82
83    pub fn with_api_key(mut self, api_key: impl Into<String>) -> Self {
84        self.api_key = Some(api_key.into());
85        self
86    }
87
88    pub fn with_endpoint(mut self, endpoint: impl Into<String>) -> Self {
89        self.api_endpoint = Some(endpoint.into());
90        self
91    }
92
93    pub fn with_organization(mut self, organization: impl Into<String>) -> Self {
94        self.organization = Some(organization.into());
95        self
96    }
97
98    pub fn with_proxy(mut self, proxy: impl Into<String>) -> Self {
99        self.proxy = Some(proxy.into());
100        self
101    }
102
103    pub fn with_timeout(mut self, timeout: u64) -> Self {
104        self.timeout = Some(timeout);
105        self
106    }
107
108    pub fn with_header(mut self, key: impl Into<String>, value: impl Into<String>) -> Self {
109        let headers = self.headers.get_or_insert_with(HeaderMap::new);
110        headers.insert(
111            HeaderName::from_bytes(key.into().as_bytes()).expect("Invalid header name"),
112            HeaderValue::from_str(&value.into()).expect("Invalid header value"),
113        );
114        self
115    }
116
117    pub fn build(self) -> Result<OpenAIClient, Box<dyn Error>> {
118        let api_endpoint = self.api_endpoint.unwrap_or_else(|| {
119            std::env::var("OPENAI_API_BASE").unwrap_or_else(|_| API_URL_V1.to_owned())
120        });
121
122        Ok(OpenAIClient {
123            api_endpoint,
124            api_key: self.api_key,
125            organization: self.organization,
126            proxy: self.proxy,
127            timeout: self.timeout,
128            headers: self.headers,
129            response_headers: None,
130        })
131    }
132}
133
134impl OpenAIClient {
135    pub fn builder() -> OpenAIClientBuilder {
136        OpenAIClientBuilder::new()
137    }
138
139    async fn build_request(&self, method: Method, path: &str) -> reqwest::RequestBuilder {
140        let url = self
141            .build_url_with_preserved_query(path)
142            .unwrap_or_else(|_| format!("{}/{}", self.api_endpoint, path));
143
144        let client = Client::builder();
145
146        #[cfg(feature = "rustls")]
147        let client = client.use_rustls_tls();
148
149        let client = if let Some(timeout) = self.timeout {
150            client.timeout(std::time::Duration::from_secs(timeout))
151        } else {
152            client
153        };
154
155        let client = if let Some(proxy) = &self.proxy {
156            client.proxy(reqwest::Proxy::all(proxy).unwrap())
157        } else {
158            client
159        };
160
161        let client = client.build().unwrap();
162
163        let mut request = client.request(method, url);
164
165        if let Some(api_key) = &self.api_key {
166            request = request.header("Authorization", format!("Bearer {}", api_key));
167        }
168
169        if let Some(organization) = &self.organization {
170            request = request.header("openai-organization", organization);
171        }
172
173        if let Some(headers) = &self.headers {
174            for (key, value) in headers {
175                request = request.header(key, value);
176            }
177        }
178
179        if Self::is_beta(path) {
180            request = request.header("OpenAI-Beta", "assistants=v2");
181        }
182
183        request
184    }
185
186    async fn post<T: serde::de::DeserializeOwned>(
187        &mut self,
188        path: &str,
189        body: &impl serde::ser::Serialize,
190    ) -> Result<T, APIError> {
191        let request = self.build_request(Method::POST, path).await;
192        let request = request.json(body);
193        let response = request.send().await?;
194        self.handle_response(response).await
195    }
196
197    async fn get<T: serde::de::DeserializeOwned>(&mut self, path: &str) -> Result<T, APIError> {
198        let request = self.build_request(Method::GET, path).await;
199        let response = request.send().await?;
200        self.handle_response(response).await
201    }
202
203    async fn get_raw(&self, path: &str) -> Result<Bytes, APIError> {
204        let request = self.build_request(Method::GET, path).await;
205        let response = request.send().await?;
206        Ok(response.bytes().await?)
207    }
208
209    async fn delete<T: serde::de::DeserializeOwned>(&mut self, path: &str) -> Result<T, APIError> {
210        let request = self.build_request(Method::DELETE, path).await;
211        let response = request.send().await?;
212        self.handle_response(response).await
213    }
214
215    async fn post_form<T: serde::de::DeserializeOwned>(
216        &mut self,
217        path: &str,
218        form: Form,
219    ) -> Result<T, APIError> {
220        let request = self.build_request(Method::POST, path).await;
221        let request = request.multipart(form);
222        let response = request.send().await?;
223        self.handle_response(response).await
224    }
225
226    async fn post_form_raw(&self, path: &str, form: Form) -> Result<Bytes, APIError> {
227        let request = self.build_request(Method::POST, path).await;
228        let request = request.multipart(form);
229        let response = request.send().await?;
230        Ok(response.bytes().await?)
231    }
232
233    async fn handle_response<T: serde::de::DeserializeOwned>(
234        &mut self,
235        response: Response,
236    ) -> Result<T, APIError> {
237        let status = response.status();
238        let headers = response.headers().clone();
239        if status.is_success() {
240            let text = response.text().await.unwrap_or_else(|_| "".to_string());
241            match serde_json::from_str::<T>(&text) {
242                Ok(parsed) => {
243                    self.response_headers = Some(headers);
244                    Ok(parsed)
245                }
246                Err(e) => Err(APIError::CustomError {
247                    message: format!("Failed to parse JSON: {} / response {}", e, text),
248                }),
249            }
250        } else {
251            let error_message = response
252                .text()
253                .await
254                .unwrap_or_else(|_| "Unknown error".to_string());
255            Err(APIError::CustomError {
256                message: format!("{}: {}", status, error_message),
257            })
258        }
259    }
260
261    pub async fn completion(
262        &mut self,
263        req: CompletionRequest,
264    ) -> Result<CompletionResponse, APIError> {
265        self.post("completions", &req).await
266    }
267
268    pub async fn edit(&mut self, req: EditRequest) -> Result<EditResponse, APIError> {
269        self.post("edits", &req).await
270    }
271
272    pub async fn image_generation(
273        &mut self,
274        req: ImageGenerationRequest,
275    ) -> Result<ImageGenerationResponse, APIError> {
276        self.post("images/generations", &req).await
277    }
278
279    pub async fn image_edit(
280        &mut self,
281        req: ImageEditRequest,
282    ) -> Result<ImageEditResponse, APIError> {
283        self.post("images/edits", &req).await
284    }
285
286    pub async fn image_variation(
287        &mut self,
288        req: ImageVariationRequest,
289    ) -> Result<ImageVariationResponse, APIError> {
290        self.post("images/variations", &req).await
291    }
292
293    pub async fn embedding(
294        &mut self,
295        req: EmbeddingRequest,
296    ) -> Result<EmbeddingResponse, APIError> {
297        self.post("embeddings", &req).await
298    }
299
300    pub async fn file_list(&mut self) -> Result<FileListResponse, APIError> {
301        self.get("files").await
302    }
303
304    pub async fn upload_file(
305        &mut self,
306        req: FileUploadRequest,
307    ) -> Result<FileUploadResponse, APIError> {
308        let form = Self::create_form(&req, "file")?;
309        self.post_form("files", form).await
310    }
311
312    pub async fn delete_file(
313        &mut self,
314        req: FileDeleteRequest,
315    ) -> Result<FileDeleteResponse, APIError> {
316        self.delete(&format!("files/{}", req.file_id)).await
317    }
318
319    pub async fn retrieve_file(
320        &mut self,
321        file_id: String,
322    ) -> Result<FileRetrieveResponse, APIError> {
323        self.get(&format!("files/{}", file_id)).await
324    }
325
326    pub async fn retrieve_file_content(&self, file_id: String) -> Result<Bytes, APIError> {
327        self.get_raw(&format!("files/{}/content", file_id)).await
328    }
329
330    pub async fn chat_completion(
331        &mut self,
332        req: ChatCompletionRequest,
333    ) -> Result<ChatCompletionResponse, APIError> {
334        self.post("chat/completions", &req).await
335    }
336
337    pub async fn audio_transcription(
338        &mut self,
339        req: AudioTranscriptionRequest,
340    ) -> Result<AudioTranscriptionResponse, APIError> {
341        // https://platform.openai.com/docs/api-reference/audio/createTranscription#audio-createtranscription-response_format
342        if let Some(response_format) = &req.response_format {
343            if response_format != "json" && response_format != "verbose_json" {
344                return Err(APIError::CustomError {
345                    message: "response_format must be either 'json' or 'verbose_json' please use audio_transcription_raw".to_string(),
346                });
347            }
348        }
349        let form: Form;
350        if req.clone().file.is_some() {
351            form = Self::create_form(&req, "file")?;
352        } else if let Some(bytes) = req.clone().bytes {
353            form = Self::create_form_from_bytes(&req, bytes)?;
354        } else {
355            return Err(APIError::CustomError {
356                message: "Either file or bytes must be provided".to_string(),
357            });
358        }
359        self.post_form("audio/transcriptions", form).await
360    }
361
362    pub async fn audio_transcription_raw(
363        &mut self,
364        req: AudioTranscriptionRequest,
365    ) -> Result<Bytes, APIError> {
366        // https://platform.openai.com/docs/api-reference/audio/createTranscription#audio-createtranscription-response_format
367        if let Some(response_format) = &req.response_format {
368            if response_format != "text" && response_format != "srt" && response_format != "vtt" {
369                return Err(APIError::CustomError {
370                    message: "response_format must be either 'text', 'srt' or 'vtt', please use audio_transcription".to_string(),
371                });
372            }
373        }
374        let form: Form;
375        if req.clone().file.is_some() {
376            form = Self::create_form(&req, "file")?;
377        } else if let Some(bytes) = req.clone().bytes {
378            form = Self::create_form_from_bytes(&req, bytes)?;
379        } else {
380            return Err(APIError::CustomError {
381                message: "Either file or bytes must be provided".to_string(),
382            });
383        }
384        self.post_form_raw("audio/transcriptions", form).await
385    }
386
387    pub async fn audio_translation(
388        &mut self,
389        req: AudioTranslationRequest,
390    ) -> Result<AudioTranslationResponse, APIError> {
391        let form = Self::create_form(&req, "file")?;
392        self.post_form("audio/translations", form).await
393    }
394
395    pub async fn audio_speech(
396        &mut self,
397        req: AudioSpeechRequest,
398    ) -> Result<AudioSpeechResponse, APIError> {
399        let request = self.build_request(Method::POST, "audio/speech").await;
400        let request = request.json(&req);
401        let response = request.send().await?;
402        let headers = response.headers().clone();
403        let bytes = response.bytes().await?;
404        let path = Path::new(req.output.as_str());
405        if let Some(parent) = path.parent() {
406            match create_dir_all(parent) {
407                Ok(_) => {}
408                Err(e) => {
409                    return Err(APIError::CustomError {
410                        message: e.to_string(),
411                    })
412                }
413            }
414        }
415        match File::create(path) {
416            Ok(mut file) => match file.write_all(&bytes) {
417                Ok(_) => {}
418                Err(e) => {
419                    return Err(APIError::CustomError {
420                        message: e.to_string(),
421                    })
422                }
423            },
424            Err(e) => {
425                return Err(APIError::CustomError {
426                    message: e.to_string(),
427                })
428            }
429        }
430
431        Ok(AudioSpeechResponse {
432            result: true,
433            headers: Some(headers),
434        })
435    }
436
437    pub async fn create_fine_tuning_job(
438        &mut self,
439        req: CreateFineTuningJobRequest,
440    ) -> Result<FineTuningJobObject, APIError> {
441        self.post("fine_tuning/jobs", &req).await
442    }
443
444    pub async fn list_fine_tuning_jobs(
445        &mut self,
446    ) -> Result<FineTuningPagination<FineTuningJobObject>, APIError> {
447        self.get("fine_tuning/jobs").await
448    }
449
450    pub async fn list_fine_tuning_job_events(
451        &mut self,
452        req: ListFineTuningJobEventsRequest,
453    ) -> Result<FineTuningPagination<FineTuningJobEvent>, APIError> {
454        self.get(&format!(
455            "fine_tuning/jobs/{}/events",
456            req.fine_tuning_job_id
457        ))
458        .await
459    }
460
461    pub async fn retrieve_fine_tuning_job(
462        &mut self,
463        req: RetrieveFineTuningJobRequest,
464    ) -> Result<FineTuningJobObject, APIError> {
465        self.get(&format!("fine_tuning/jobs/{}", req.fine_tuning_job_id))
466            .await
467    }
468
469    pub async fn cancel_fine_tuning_job(
470        &mut self,
471        req: CancelFineTuningJobRequest,
472    ) -> Result<FineTuningJobObject, APIError> {
473        self.post(
474            &format!("fine_tuning/jobs/{}/cancel", req.fine_tuning_job_id),
475            &req,
476        )
477        .await
478    }
479
480    pub async fn create_moderation(
481        &mut self,
482        req: CreateModerationRequest,
483    ) -> Result<CreateModerationResponse, APIError> {
484        self.post("moderations", &req).await
485    }
486
487    pub async fn create_assistant(
488        &mut self,
489        req: AssistantRequest,
490    ) -> Result<AssistantObject, APIError> {
491        self.post("assistants", &req).await
492    }
493
494    pub async fn retrieve_assistant(
495        &mut self,
496        assistant_id: String,
497    ) -> Result<AssistantObject, APIError> {
498        self.get(&format!("assistants/{}", assistant_id)).await
499    }
500
501    pub async fn modify_assistant(
502        &mut self,
503        assistant_id: String,
504        req: AssistantRequest,
505    ) -> Result<AssistantObject, APIError> {
506        self.post(&format!("assistants/{}", assistant_id), &req)
507            .await
508    }
509
510    pub async fn delete_assistant(
511        &mut self,
512        assistant_id: String,
513    ) -> Result<common::DeletionStatus, APIError> {
514        self.delete(&format!("assistants/{}", assistant_id)).await
515    }
516
517    pub async fn list_assistant(
518        &mut self,
519        limit: Option<i64>,
520        order: Option<String>,
521        after: Option<String>,
522        before: Option<String>,
523    ) -> Result<ListAssistant, APIError> {
524        let url = Self::query_params(limit, order, after, before, "assistants".to_string());
525        self.get(&url).await
526    }
527
528    pub async fn create_assistant_file(
529        &mut self,
530        assistant_id: String,
531        req: AssistantFileRequest,
532    ) -> Result<AssistantFileObject, APIError> {
533        self.post(&format!("assistants/{}/files", assistant_id), &req)
534            .await
535    }
536
537    pub async fn retrieve_assistant_file(
538        &mut self,
539        assistant_id: String,
540        file_id: String,
541    ) -> Result<AssistantFileObject, APIError> {
542        self.get(&format!("assistants/{}/files/{}", assistant_id, file_id))
543            .await
544    }
545
546    pub async fn delete_assistant_file(
547        &mut self,
548        assistant_id: String,
549        file_id: String,
550    ) -> Result<common::DeletionStatus, APIError> {
551        self.delete(&format!("assistants/{}/files/{}", assistant_id, file_id))
552            .await
553    }
554
555    pub async fn list_assistant_file(
556        &mut self,
557        assistant_id: String,
558        limit: Option<i64>,
559        order: Option<String>,
560        after: Option<String>,
561        before: Option<String>,
562    ) -> Result<ListAssistantFile, APIError> {
563        let url = Self::query_params(
564            limit,
565            order,
566            after,
567            before,
568            format!("assistants/{}/files", assistant_id),
569        );
570        self.get(&url).await
571    }
572
573    pub async fn create_thread(
574        &mut self,
575        req: CreateThreadRequest,
576    ) -> Result<ThreadObject, APIError> {
577        self.post("threads", &req).await
578    }
579
580    pub async fn retrieve_thread(&mut self, thread_id: String) -> Result<ThreadObject, APIError> {
581        self.get(&format!("threads/{}", thread_id)).await
582    }
583
584    pub async fn modify_thread(
585        &mut self,
586        thread_id: String,
587        req: ModifyThreadRequest,
588    ) -> Result<ThreadObject, APIError> {
589        self.post(&format!("threads/{}", thread_id), &req).await
590    }
591
592    pub async fn delete_thread(
593        &mut self,
594        thread_id: String,
595    ) -> Result<common::DeletionStatus, APIError> {
596        self.delete(&format!("threads/{}", thread_id)).await
597    }
598
599    pub async fn create_message(
600        &mut self,
601        thread_id: String,
602        req: CreateMessageRequest,
603    ) -> Result<MessageObject, APIError> {
604        self.post(&format!("threads/{}/messages", thread_id), &req)
605            .await
606    }
607
608    pub async fn retrieve_message(
609        &mut self,
610        thread_id: String,
611        message_id: String,
612    ) -> Result<MessageObject, APIError> {
613        self.get(&format!("threads/{}/messages/{}", thread_id, message_id))
614            .await
615    }
616
617    pub async fn modify_message(
618        &mut self,
619        thread_id: String,
620        message_id: String,
621        req: ModifyMessageRequest,
622    ) -> Result<MessageObject, APIError> {
623        self.post(
624            &format!("threads/{}/messages/{}", thread_id, message_id),
625            &req,
626        )
627        .await
628    }
629
630    pub async fn list_messages(&mut self, thread_id: String) -> Result<ListMessage, APIError> {
631        self.get(&format!("threads/{}/messages", thread_id)).await
632    }
633
634    pub async fn retrieve_message_file(
635        &mut self,
636        thread_id: String,
637        message_id: String,
638        file_id: String,
639    ) -> Result<MessageFileObject, APIError> {
640        self.get(&format!(
641            "threads/{}/messages/{}/files/{}",
642            thread_id, message_id, file_id
643        ))
644        .await
645    }
646
647    pub async fn list_message_file(
648        &mut self,
649        thread_id: String,
650        message_id: String,
651        limit: Option<i64>,
652        order: Option<String>,
653        after: Option<String>,
654        before: Option<String>,
655    ) -> Result<ListMessageFile, APIError> {
656        let url = Self::query_params(
657            limit,
658            order,
659            after,
660            before,
661            format!("threads/{}/messages/{}/files", thread_id, message_id),
662        );
663        self.get(&url).await
664    }
665
666    pub async fn create_run(
667        &mut self,
668        thread_id: String,
669        req: CreateRunRequest,
670    ) -> Result<RunObject, APIError> {
671        self.post(&format!("threads/{}/runs", thread_id), &req)
672            .await
673    }
674
675    pub async fn retrieve_run(
676        &mut self,
677        thread_id: String,
678        run_id: String,
679    ) -> Result<RunObject, APIError> {
680        self.get(&format!("threads/{}/runs/{}", thread_id, run_id))
681            .await
682    }
683
684    pub async fn modify_run(
685        &mut self,
686        thread_id: String,
687        run_id: String,
688        req: ModifyRunRequest,
689    ) -> Result<RunObject, APIError> {
690        self.post(&format!("threads/{}/runs/{}", thread_id, run_id), &req)
691            .await
692    }
693
694    pub async fn list_run(
695        &mut self,
696        thread_id: String,
697        limit: Option<i64>,
698        order: Option<String>,
699        after: Option<String>,
700        before: Option<String>,
701    ) -> Result<ListRun, APIError> {
702        let url = Self::query_params(
703            limit,
704            order,
705            after,
706            before,
707            format!("threads/{}/runs", thread_id),
708        );
709        self.get(&url).await
710    }
711
712    pub async fn cancel_run(
713        &mut self,
714        thread_id: String,
715        run_id: String,
716    ) -> Result<RunObject, APIError> {
717        self.post(
718            &format!("threads/{}/runs/{}/cancel", thread_id, run_id),
719            &ModifyRunRequest::default(),
720        )
721        .await
722    }
723
724    pub async fn create_thread_and_run(
725        &mut self,
726        req: CreateThreadAndRunRequest,
727    ) -> Result<RunObject, APIError> {
728        self.post("threads/runs", &req).await
729    }
730
731    pub async fn retrieve_run_step(
732        &mut self,
733        thread_id: String,
734        run_id: String,
735        step_id: String,
736    ) -> Result<RunStepObject, APIError> {
737        self.get(&format!(
738            "threads/{}/runs/{}/steps/{}",
739            thread_id, run_id, step_id
740        ))
741        .await
742    }
743
744    pub async fn list_run_step(
745        &mut self,
746        thread_id: String,
747        run_id: String,
748        limit: Option<i64>,
749        order: Option<String>,
750        after: Option<String>,
751        before: Option<String>,
752    ) -> Result<ListRunStep, APIError> {
753        let url = Self::query_params(
754            limit,
755            order,
756            after,
757            before,
758            format!("threads/{}/runs/{}/steps", thread_id, run_id),
759        );
760        self.get(&url).await
761    }
762
763    pub async fn create_batch(
764        &mut self,
765        req: CreateBatchRequest,
766    ) -> Result<BatchResponse, APIError> {
767        self.post("batches", &req).await
768    }
769
770    pub async fn retrieve_batch(&mut self, batch_id: String) -> Result<BatchResponse, APIError> {
771        self.get(&format!("batches/{}", batch_id)).await
772    }
773
774    pub async fn cancel_batch(&mut self, batch_id: String) -> Result<BatchResponse, APIError> {
775        self.post(
776            &format!("batches/{}/cancel", batch_id),
777            &common::EmptyRequestBody {},
778        )
779        .await
780    }
781
782    pub async fn list_batch(
783        &mut self,
784        after: Option<String>,
785        limit: Option<i64>,
786    ) -> Result<ListBatchResponse, APIError> {
787        let url = Self::query_params(limit, None, after, None, "batches".to_string());
788        self.get(&url).await
789    }
790
791    pub async fn list_models(&mut self) -> Result<ModelsResponse, APIError> {
792        self.get("models").await
793    }
794
795    pub async fn retrieve_model(&mut self, model_id: String) -> Result<ModelResponse, APIError> {
796        self.get(&format!("models/{}", model_id)).await
797    }
798
799    pub async fn delete_model(
800        &mut self,
801        model_id: String,
802    ) -> Result<common::DeletionStatus, APIError> {
803        self.delete(&format!("models/{}", model_id)).await
804    }
805
806    fn build_url_with_preserved_query(&self, path: &str) -> Result<String, url::ParseError> {
807        let (base, query_opt) = match self.api_endpoint.split_once('?') {
808            Some((b, q)) => (b.trim_end_matches('/'), Some(q)),
809            None => (self.api_endpoint.trim_end_matches('/'), None),
810        };
811
812        let full_path = format!("{}/{}", base, path.trim_start_matches('/'));
813        let mut url = Url::parse(&full_path)?;
814
815        if let Some(query) = query_opt {
816            for (k, v) in url::form_urlencoded::parse(query.as_bytes()) {
817                url.query_pairs_mut().append_pair(&k, &v);
818            }
819        }
820        Ok(url.to_string())
821    }
822
823    fn query_params(
824        limit: Option<i64>,
825        order: Option<String>,
826        after: Option<String>,
827        before: Option<String>,
828        mut url: String,
829    ) -> String {
830        let mut params = vec![];
831        if let Some(limit) = limit {
832            params.push(format!("limit={}", limit));
833        }
834        if let Some(order) = order {
835            params.push(format!("order={}", order));
836        }
837        if let Some(after) = after {
838            params.push(format!("after={}", after));
839        }
840        if let Some(before) = before {
841            params.push(format!("before={}", before));
842        }
843        if !params.is_empty() {
844            url = format!("{}?{}", url, params.join("&"));
845        }
846        url
847    }
848
849    fn is_beta(path: &str) -> bool {
850        path.starts_with("assistants") || path.starts_with("threads")
851    }
852
853    fn create_form<T>(req: &T, file_field: &str) -> Result<Form, APIError>
854    where
855        T: Serialize,
856    {
857        let json = match serde_json::to_value(req) {
858            Ok(json) => json,
859            Err(e) => {
860                return Err(APIError::CustomError {
861                    message: e.to_string(),
862                })
863            }
864        };
865        let file_path = if let Value::Object(map) = &json {
866            map.get(file_field)
867                .and_then(|v| v.as_str())
868                .ok_or(APIError::CustomError {
869                    message: format!("Field '{}' not found or not a string", file_field),
870                })?
871        } else {
872            return Err(APIError::CustomError {
873                message: "Request is not a JSON object".to_string(),
874            });
875        };
876
877        let mut file = match File::open(file_path) {
878            Ok(file) => file,
879            Err(e) => {
880                return Err(APIError::CustomError {
881                    message: e.to_string(),
882                })
883            }
884        };
885        let mut buffer = Vec::new();
886        match file.read_to_end(&mut buffer) {
887            Ok(_) => {}
888            Err(e) => {
889                return Err(APIError::CustomError {
890                    message: e.to_string(),
891                })
892            }
893        }
894
895        let mut form =
896            Form::new().part("file", Part::bytes(buffer).file_name(file_path.to_string()));
897
898        if let Value::Object(map) = json {
899            for (key, value) in map.into_iter() {
900                if key != file_field {
901                    match value {
902                        Value::String(s) => {
903                            form = form.text(key, s);
904                        }
905                        Value::Number(n) => {
906                            form = form.text(key, n.to_string());
907                        }
908                        _ => {}
909                    }
910                }
911            }
912        }
913
914        Ok(form)
915    }
916
917    fn create_form_from_bytes<T>(req: &T, bytes: Vec<u8>) -> Result<Form, APIError>
918    where
919        T: Serialize,
920    {
921        let json = match serde_json::to_value(req) {
922            Ok(json) => json,
923            Err(e) => {
924                return Err(APIError::CustomError {
925                    message: e.to_string(),
926                })
927            }
928        };
929
930        let mut form = Form::new().part("file", Part::bytes(bytes.clone()).file_name("file.mp3"));
931
932        if let Value::Object(map) = json {
933            for (key, value) in map.into_iter() {
934                match value {
935                    Value::String(s) => {
936                        form = form.text(key, s);
937                    }
938                    Value::Number(n) => {
939                        form = form.text(key, n.to_string());
940                    }
941                    _ => {}
942                }
943            }
944        }
945
946        Ok(form)
947    }
948}