1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
pub mod config;
pub mod model;

use super::client::Client;
use super::entry_point::{EntryPoint, Function, Version};
use anyhow::Result;
use config::ChatConfig;
use model::{CompletionRequest, CompletionResponse, Message};

#[derive(Debug, Default, Clone, PartialEq)]
pub struct ChatBuilder {
    version: Option<Version>,
    config: Option<ChatConfig>,
    request: Option<Vec<Message>>,
}

impl ChatBuilder {
    pub fn version(mut self, value: Version) -> Self {
        self.version = Some(value);
        self
    }
    pub fn config(mut self, value: ChatConfig) -> Self {
        self.config = Some(value);
        self
    }
    pub fn request(mut self, value: Vec<Message>) -> Self {
        self.request = Some(value);
        self
    }
    pub fn build(&self) -> Result<Chat> {
        let version = self.version.unwrap_or(Default::default());
        let config = match self.config.clone() {
            Some(val) => val,
            _ => return Err(anyhow::anyhow!("Config must be set.")),
        };
        let entry_point = EntryPoint::default()
            .set_version(version)
            .set_function(Function::CreateChatCompletion);
        let request = match &self.request {
            Some(val) => val.clone(),
            _ => return Err(anyhow::anyhow!("Request must be set.")),
        };
        Ok(Chat {
            entry_point,
            config,
            request,
        })
    }
}

#[derive(Debug, Clone, PartialEq)]
pub struct Chat {
    entry_point: EntryPoint,
    config: ChatConfig,
    request: Vec<Message>,
}

impl Chat {
    pub fn new(config: ChatConfig, request: Vec<Message>) -> Self {
        Self {
            entry_point: Default::default(),
            config,
            request,
        }
    }
    pub fn builder() -> ChatBuilder {
        Default::default()
    }
    pub async fn execute(&self, client: &Client) -> Result<CompletionResponse> {
        let request = CompletionRequest {
            model: self.config.model.as_ref().to_string(),
            temperature: self.config.temperature,
            messages: self.request.clone(),
        };
        let res = client.post(&self.entry_point.path(), request).await?;
        let res: CompletionResponse = serde_json::from_str(res.text().await?.as_str())?;
        Ok(res)
    }
}

#[cfg(test)]
mod tests {
    use super::super::client::Config;
    use super::*;
    #[test]
    fn builder() -> Result<()> {
        let builder = ChatBuilder::default();
        assert!(builder.clone().build().is_err());
        let builder = builder.config(ChatConfig::default()).request(vec![]);
        assert!(builder.build().is_ok());
        Ok(())
    }

    #[tokio::test]
    async fn post() -> Result<()> {
        let client = Client::new(Config::from_env()?)?;
        let res = Chat::builder()
            .config(Default::default())
            .request(vec!["hi".into()])
            .build()?
            .execute(&client)
            .await;
        assert!(res.is_ok());
        Ok(())
    }
}