supabase_client_query/
csv_select.rs1use reqwest::header::HeaderValue;
2
3use supabase_client_core::SupabaseError;
4
5use crate::backend::QueryBackend;
6use crate::filter::Filterable;
7use crate::modifier::Modifiable;
8use crate::sql::{FilterCondition, ParamStore, SqlParts};
9
10pub struct CsvSelectBuilder {
15 pub(crate) backend: QueryBackend,
16 pub(crate) parts: SqlParts,
17 pub(crate) params: ParamStore,
18}
19
20impl Filterable for CsvSelectBuilder {
21 fn filters_mut(&mut self) -> &mut Vec<FilterCondition> {
22 &mut self.parts.filters
23 }
24 fn params_mut(&mut self) -> &mut ParamStore {
25 &mut self.params
26 }
27}
28
29impl Modifiable for CsvSelectBuilder {
30 fn parts_mut(&mut self) -> &mut SqlParts {
31 &mut self.parts
32 }
33}
34
35impl CsvSelectBuilder {
36 pub fn schema(mut self, schema: &str) -> Self {
38 self.parts.schema_override = Some(schema.to_string());
39 self
40 }
41
42 pub async fn execute(self) -> Result<String, SupabaseError> {
44 match &self.backend {
45 QueryBackend::Rest { http, base_url, api_key, schema } => {
46 let (url, mut headers) = crate::postgrest::build_postgrest_select(
47 base_url, &self.parts, &self.params,
48 )
49 .map_err(SupabaseError::QueryBuilder)?;
50
51 headers.insert("Accept", HeaderValue::from_static("text/csv"));
53
54 headers.insert("apikey", HeaderValue::from_str(api_key).unwrap());
56 headers.insert(
57 "Authorization",
58 HeaderValue::from_str(&format!("Bearer {}", api_key)).unwrap(),
59 );
60
61 if let Some(ref so) = self.parts.schema_override {
63 headers.insert(
64 "Accept-Profile",
65 HeaderValue::from_str(so).unwrap(),
66 );
67 } else if schema != "public" {
68 headers.entry("Accept-Profile")
69 .or_insert_with(|| HeaderValue::from_str(schema).unwrap());
70 }
71
72 let response = http
73 .get(&url)
74 .headers(headers)
75 .send()
76 .await
77 .map_err(|e| SupabaseError::Http(e.to_string()))?;
78
79 let status = response.status().as_u16();
80 let body = response
81 .text()
82 .await
83 .map_err(|e| SupabaseError::Http(e.to_string()))?;
84
85 if status >= 400 {
86 return Err(SupabaseError::postgrest(status, body, None));
87 }
88
89 Ok(body)
90 }
91 #[cfg(feature = "direct-sql")]
92 QueryBackend::DirectSql { .. } => {
93 Err(SupabaseError::query_builder(
94 "CSV output is only supported with the REST (PostgREST) backend",
95 ))
96 }
97 }
98 }
99}
100
101#[cfg(test)]
102mod tests {
103 use super::*;
104 use crate::sql::{SqlOperation, SqlParts, ParamStore};
105
106 #[test]
107 fn test_csv_builder_modifiable() {
108 let mut builder = CsvSelectBuilder {
109 backend: QueryBackend::Rest {
110 http: reqwest::Client::new(),
111 base_url: "http://localhost".into(),
112 api_key: "key".into(),
113 schema: "public".to_string(),
114 },
115 parts: SqlParts::new(SqlOperation::Select, "public", "cities"),
116 params: ParamStore::new(),
117 };
118 builder = builder.limit(10);
119 assert_eq!(builder.parts.limit, Some(10));
120 }
121
122 #[test]
123 fn test_csv_builder_filterable() {
124 let builder = CsvSelectBuilder {
125 backend: QueryBackend::Rest {
126 http: reqwest::Client::new(),
127 base_url: "http://localhost".into(),
128 api_key: "key".into(),
129 schema: "public".to_string(),
130 },
131 parts: SqlParts::new(SqlOperation::Select, "public", "cities"),
132 params: ParamStore::new(),
133 };
134 let builder = builder.eq("name", "Auckland");
135 assert_eq!(builder.parts.filters.len(), 1);
136 }
137
138 #[test]
139 fn test_csv_accept_header() {
140 let parts = SqlParts::new(SqlOperation::Select, "public", "cities");
141 let params = ParamStore::new();
142 let (_, mut headers) = crate::postgrest::build_postgrest_select(
143 "http://localhost:64321", &parts, ¶ms,
144 ).unwrap();
145 headers.insert("Accept", HeaderValue::from_static("text/csv"));
147 assert_eq!(headers.get("Accept").unwrap(), "text/csv");
148 }
149}