use super::BETA_HEADER;
use crate::client::OpenAI;
use crate::error::OpenAIError;
use crate::pagination::{Page, Paginator};
use crate::types::beta::{
Assistant, AssistantCreateRequest, AssistantDeleted, AssistantList, AssistantListParams,
};
pub struct Assistants<'a> {
client: &'a OpenAI,
}
impl<'a> Assistants<'a> {
pub(crate) fn new(client: &'a OpenAI) -> Self {
Self { client }
}
pub async fn create(&self, request: AssistantCreateRequest) -> Result<Assistant, OpenAIError> {
let response = self
.client
.request(reqwest::Method::POST, "/assistants")
.header(BETA_HEADER.0, BETA_HEADER.1)
.json(&request)
.send()
.await?;
OpenAI::handle_response(response).await
}
pub async fn list(&self) -> Result<AssistantList, OpenAIError> {
let response = self
.client
.request(reqwest::Method::GET, "/assistants")
.header(BETA_HEADER.0, BETA_HEADER.1)
.send()
.await?;
OpenAI::handle_response(response).await
}
pub async fn list_page(
&self,
params: AssistantListParams,
) -> Result<AssistantList, OpenAIError> {
let response = self
.client
.request(reqwest::Method::GET, "/assistants")
.header(BETA_HEADER.0, BETA_HEADER.1)
.query(¶ms.to_query())
.send()
.await?;
OpenAI::handle_response(response).await
}
pub fn list_auto(&self, params: AssistantListParams) -> Paginator<Assistant> {
let client = self.client.clone();
let base_params = params;
Paginator::new(move |cursor| {
let client = client.clone();
let mut params = base_params.clone();
if cursor.is_some() {
params.after = cursor;
}
async move {
let response = client
.request(reqwest::Method::GET, "/assistants")
.header(BETA_HEADER.0, BETA_HEADER.1)
.query(¶ms.to_query())
.send()
.await?;
let list: AssistantList = OpenAI::handle_response(response).await?;
let after_cursor = list
.last_id
.clone()
.or_else(|| list.data.last().map(|a| a.id.clone()));
Ok(Page {
has_more: list.has_more.unwrap_or(false),
after_cursor,
data: list.data,
})
}
})
}
pub async fn retrieve(&self, assistant_id: &str) -> Result<Assistant, OpenAIError> {
let response = self
.client
.request(reqwest::Method::GET, &format!("/assistants/{assistant_id}"))
.header(BETA_HEADER.0, BETA_HEADER.1)
.send()
.await?;
OpenAI::handle_response(response).await
}
pub async fn delete(&self, assistant_id: &str) -> Result<AssistantDeleted, OpenAIError> {
let response = self
.client
.request(
reqwest::Method::DELETE,
&format!("/assistants/{assistant_id}"),
)
.header(BETA_HEADER.0, BETA_HEADER.1)
.send()
.await?;
OpenAI::handle_response(response).await
}
}
#[cfg(test)]
mod tests {
use crate::OpenAI;
use crate::config::ClientConfig;
use crate::types::beta::AssistantCreateRequest;
#[tokio::test]
async fn test_assistants_create() {
let mut server = mockito::Server::new_async().await;
let mock = server
.mock("POST", "/assistants")
.match_header("OpenAI-Beta", "assistants=v2")
.with_status(200)
.with_header("content-type", "application/json")
.with_body(
r#"{
"id": "asst_abc123",
"object": "assistant",
"created_at": 1699009709,
"model": "gpt-4o",
"name": "Math Tutor",
"tools": [{"type": "code_interpreter"}]
}"#,
)
.create_async()
.await;
let client = OpenAI::with_config(ClientConfig::new("sk-test").base_url(server.url()));
let mut request = AssistantCreateRequest::new("gpt-4o");
request.name = Some("Math Tutor".into());
let assistant = client.beta().assistants().create(request).await.unwrap();
assert_eq!(assistant.id, "asst_abc123");
assert_eq!(assistant.name.as_deref(), Some("Math Tutor"));
mock.assert_async().await;
}
#[tokio::test]
async fn test_assistants_delete() {
let mut server = mockito::Server::new_async().await;
let mock = server
.mock("DELETE", "/assistants/asst_abc123")
.match_header("OpenAI-Beta", "assistants=v2")
.with_status(200)
.with_header("content-type", "application/json")
.with_body(r#"{"id": "asst_abc123", "object": "assistant.deleted", "deleted": true}"#)
.create_async()
.await;
let client = OpenAI::with_config(ClientConfig::new("sk-test").base_url(server.url()));
let resp = client
.beta()
.assistants()
.delete("asst_abc123")
.await
.unwrap();
assert!(resp.deleted);
mock.assert_async().await;
}
}