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
use core::fmt;

use crate::{structs::ExecuteResponse, Deserializer, PSConnection};
use anyhow::Result;

pub struct QueryBuilder {
    query: String,
    values: Vec<String>,
}

impl QueryBuilder {
    pub fn new(query: &str) -> Self {
        Self {
            query: query.to_string(),
            values: Vec::new(),
        }
    }

    pub fn bind<T: ToString>(mut self, value: T) -> Self {
        let sanitized = value
            .to_string()
            .replace("'", "''")
            .replace("\"", "\\\"")
            .replace("`", "\\`");

        self.values.push(sanitized);
        self
    }

    fn generate_query(&self) -> String {
        let mut query = self.query.clone();
        for i in 0..self.values.len() {
            query = query.replace(&format!("${}", i), &self.values[i]);
        }

        query
    }

    pub async fn execute(self, connection: &mut PSConnection) -> Result<()> {
        connection.execute(&self.generate_query()).await
    }

    pub async fn execute_raw(self, connection: &mut PSConnection) -> Result<ExecuteResponse> {
        connection.execute_raw(&self.generate_query()).await
    }

    pub async fn fetch_one<T>(self, conn: &mut PSConnection) -> Result<T>
    where
        T: Deserializer,
    {
        let res = self.execute_raw(conn).await?;
        if let Some(err) = res.error {
            anyhow::bail!("Code: \"{}\", message: \"{}\"", err.code, err.message);
        }

        let res = res.deserialize()?;
        Ok(res)
    }

    pub async fn fetch_all<T>(self, conn: &mut PSConnection) -> Result<Vec<T>>
    where
        T: Deserializer,
    {
        let res = self.execute_raw(conn).await?;
        if let Some(err) = res.error {
            anyhow::bail!("Code: \"{}\", message: \"{}\"", err.code, err.message);
        }

        let res = res.deserialize_multiple()?;
        Ok(res)
    }

    fn sql(&self) -> String {
        let mut query = self.query.clone();
        for i in 0..self.values.len() {
            query = query.replace(&format!("${}", i), &self.values[i]);
        }

        query
    }
}

impl fmt::Debug for QueryBuilder {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        write!(f, "{}", self.sql())
    }
}