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
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
#![allow(unused_imports)]

use std::str::FromStr;

#[cfg(not(target_arch = "wasm32"))]
use napi::bindgen_prelude::{FromNapiValue, ToNapiValue};

use quaint::connector::{ExternalConnectionInfo, SqlFamily};
#[cfg(target_arch = "wasm32")]
use tsify::Tsify;

use crate::conversion::JSArg;
use serde::{Deserialize, Serialize};
use serde_repr::{Deserialize_repr, Serialize_repr};

#[cfg_attr(target_arch = "wasm32", derive(Deserialize))]
#[derive(Debug, Eq, PartialEq, Clone)]
pub enum AdapterFlavour {
    #[cfg(feature = "mysql")]
    Mysql,
    #[cfg(feature = "postgresql")]
    Postgres,
    #[cfg(feature = "sqlite")]
    Sqlite,
}

impl FromStr for AdapterFlavour {
    type Err = String;

    fn from_str(s: &str) -> Result<Self, Self::Err> {
        match s {
            #[cfg(feature = "postgresql")]
            "postgres" => Ok(Self::Postgres),
            #[cfg(feature = "mysql")]
            "mysql" => Ok(Self::Mysql),
            #[cfg(feature = "sqlite")]
            "sqlite" => Ok(Self::Sqlite),
            _ => Err(format!("Unsupported adapter flavour: {:?}", s)),
        }
    }
}

impl From<AdapterFlavour> for SqlFamily {
    fn from(value: AdapterFlavour) -> Self {
        match value {
            #[cfg(feature = "mysql")]
            AdapterFlavour::Mysql => SqlFamily::Mysql,
            #[cfg(feature = "postgresql")]
            AdapterFlavour::Postgres => SqlFamily::Postgres,
            #[cfg(feature = "sqlite")]
            AdapterFlavour::Sqlite => SqlFamily::Sqlite,
        }
    }
}

#[cfg_attr(not(target_arch = "wasm32"), napi_derive::napi(object))]
#[cfg_attr(target_arch = "wasm32", derive(Deserialize))]
#[cfg_attr(target_arch = "wasm32", serde(rename_all = "camelCase"))]
#[derive(Default)]
pub(crate) struct JsConnectionInfo {
    pub schema_name: Option<String>,
}

impl JsConnectionInfo {
    pub fn into_external_connection_info(self, provider: &AdapterFlavour) -> ExternalConnectionInfo {
        let schema_name = self.get_schema_name(provider);
        let sql_family = provider.to_owned().into();
        ExternalConnectionInfo::new(sql_family, schema_name.to_owned())
    }
    fn get_schema_name(&self, provider: &AdapterFlavour) -> &str {
        match self.schema_name.as_ref() {
            Some(name) => name,
            None => self.default_schema_name(provider),
        }
    }

    fn default_schema_name(&self, provider: &AdapterFlavour) -> &str {
        match provider {
            #[cfg(feature = "mysql")]
            AdapterFlavour::Mysql => quaint::connector::DEFAULT_MYSQL_DB,
            #[cfg(feature = "postgresql")]
            AdapterFlavour::Postgres => quaint::connector::DEFAULT_POSTGRES_SCHEMA,
            #[cfg(feature = "sqlite")]
            AdapterFlavour::Sqlite => quaint::connector::DEFAULT_SQLITE_DATABASE,
        }
    }
}

/// This result set is more convenient to be manipulated from both Rust and NodeJS.
/// Quaint's version of ResultSet is:
///
/// pub struct ResultSet {
///     pub(crate) columns: Arc<Vec<String>>,
///     pub(crate) rows: Vec<Vec<Value<'static>>>,
///     pub(crate) last_insert_id: Option<u64>,
/// }
///
/// If we used this ResultSet would we would have worse ergonomics as quaint::Value is a structured
/// enum and cannot be used directly with the #[napi(Object)] macro. Thus requiring us to implement
/// the FromNapiValue and ToNapiValue traits for quaint::Value, and use a different custom type
/// representing the Value in javascript.
///
#[cfg_attr(not(target_arch = "wasm32"), napi_derive::napi(object))]
#[cfg_attr(target_arch = "wasm32", derive(Deserialize))]
#[cfg_attr(target_arch = "wasm32", serde(rename_all = "camelCase"))]
#[derive(Debug)]
pub struct JSResultSet {
    pub column_types: Vec<ColumnType>,
    pub column_names: Vec<String>,
    // Note this might be encoded differently for performance reasons
    pub rows: Vec<Vec<serde_json::Value>>,
    pub last_insert_id: Option<String>,
}

impl JSResultSet {
    pub fn len(&self) -> usize {
        self.rows.len()
    }
}

#[cfg_attr(not(target_arch = "wasm32"), napi_derive::napi)]
#[cfg_attr(target_arch = "wasm32", derive(Clone, Copy, Deserialize_repr))]
#[repr(u8)]
#[derive(Debug)]
pub enum ColumnType {
    // [PLANETSCALE_TYPE] (MYSQL_TYPE) -> [TypeScript example]
    /// The following PlanetScale type IDs are mapped into Int32:
    /// - INT8 (TINYINT) -> e.g. `127`
    /// - INT16 (SMALLINT) -> e.g. `32767`
    /// - INT24 (MEDIUMINT) -> e.g. `8388607`
    /// - INT32 (INT) -> e.g. `2147483647`
    Int32 = 0,

    /// The following PlanetScale type IDs are mapped into Int64:
    /// - INT64 (BIGINT) -> e.g. `"9223372036854775807"` (String-encoded)
    Int64 = 1,

    /// The following PlanetScale type IDs are mapped into Float:
    /// - FLOAT32 (FLOAT) -> e.g. `3.402823466`
    Float = 2,

    /// The following PlanetScale type IDs are mapped into Double:
    /// - FLOAT64 (DOUBLE) -> e.g. `1.7976931348623157`
    Double = 3,

    /// The following PlanetScale type IDs are mapped into Numeric:
    /// - DECIMAL (DECIMAL) -> e.g. `"99999999.99"` (String-encoded)
    Numeric = 4,

    /// The following PlanetScale type IDs are mapped into Boolean:
    /// - BOOLEAN (BOOLEAN) -> e.g. `1`
    Boolean = 5,

    Character = 6,

    /// The following PlanetScale type IDs are mapped into Text:
    /// - TEXT (TEXT) -> e.g. `"foo"` (String-encoded)
    /// - VARCHAR (VARCHAR) -> e.g. `"foo"` (String-encoded)
    Text = 7,

    /// The following PlanetScale type IDs are mapped into Date:
    /// - DATE (DATE) -> e.g. `"2023-01-01"` (String-encoded, yyyy-MM-dd)
    Date = 8,

    /// The following PlanetScale type IDs are mapped into Time:
    /// - TIME (TIME) -> e.g. `"23:59:59"` (String-encoded, HH:mm:ss)
    Time = 9,

    /// The following PlanetScale type IDs are mapped into DateTime:
    /// - DATETIME (DATETIME) -> e.g. `"2023-01-01 23:59:59"` (String-encoded, yyyy-MM-dd HH:mm:ss)
    /// - TIMESTAMP (TIMESTAMP) -> e.g. `"2023-01-01 23:59:59"` (String-encoded, yyyy-MM-dd HH:mm:ss)
    DateTime = 10,

    /// The following PlanetScale type IDs are mapped into Json:
    /// - JSON (JSON) -> e.g. `"{\"key\": \"value\"}"` (String-encoded)
    Json = 11,

    /// The following PlanetScale type IDs are mapped into Enum:
    /// - ENUM (ENUM) -> e.g. `"foo"` (String-encoded)
    Enum = 12,

    /// The following PlanetScale type IDs are mapped into Bytes:
    /// - BLOB (BLOB) -> e.g. `"\u0012"` (String-encoded)
    /// - VARBINARY (VARBINARY) -> e.g. `"\u0012"` (String-encoded)
    /// - BINARY (BINARY) -> e.g. `"\u0012"` (String-encoded)
    /// - GEOMETRY (GEOMETRY) -> e.g. `"\u0012"` (String-encoded)
    Bytes = 13,

    /// The following PlanetScale type IDs are mapped into Set:
    /// - SET (SET) -> e.g. `"foo,bar"` (String-encoded, comma-separated)
    /// This is currently unhandled, and will panic if encountered.
    Set = 14,

    /// UUID from postgres-flavored driver adapters is mapped to this type.
    Uuid = 15,

    /*
     * Scalar arrays
     */
    /// Int32 array (INT2_ARRAY and INT4_ARRAY in PostgreSQL)
    Int32Array = 64,

    /// Int64 array (INT8_ARRAY in PostgreSQL)
    Int64Array = 65,

    /// Float array (FLOAT4_ARRAY in PostgreSQL)
    FloatArray = 66,

    /// Double array (FLOAT8_ARRAY in PostgreSQL)
    DoubleArray = 67,

    /// Numeric array (NUMERIC_ARRAY, MONEY_ARRAY etc in PostgreSQL)
    NumericArray = 68,

    /// Boolean array (BOOL_ARRAY in PostgreSQL)
    BooleanArray = 69,

    /// Char array (CHAR_ARRAY in PostgreSQL)
    CharacterArray = 70,

    /// Text array (TEXT_ARRAY in PostgreSQL)
    TextArray = 71,

    /// Date array (DATE_ARRAY in PostgreSQL)
    DateArray = 72,

    /// Time array (TIME_ARRAY in PostgreSQL)
    TimeArray = 73,

    /// DateTime array (TIMESTAMP_ARRAY in PostgreSQL)
    DateTimeArray = 74,

    /// Json array (JSON_ARRAY in PostgreSQL)
    JsonArray = 75,

    /// Enum array
    EnumArray = 76,

    /// Bytes array (BYTEA_ARRAY in PostgreSQL)
    BytesArray = 77,

    /// Uuid array (UUID_ARRAY in PostgreSQL)
    UuidArray = 78,

    /*
     * Below there are custom types that don't have a 1:1 translation with a quaint::Value.
     * enum variant.
     */
    /// UnknownNumber is used when the type of the column is a number but of unknown particular type
    /// and precision.
    ///
    /// It's used by some driver adapters, like libsql to return aggregation values like AVG, or
    /// COUNT, and it can be mapped to either Int64, or Double
    UnknownNumber = 128,
}

#[cfg_attr(not(target_arch = "wasm32"), napi_derive::napi(object))]
#[derive(Debug, Default)]
pub struct Query {
    pub sql: String,
    pub args: Vec<JSArg>,
}

#[cfg_attr(not(target_arch = "wasm32"), napi_derive::napi(object))]
#[cfg_attr(target_arch = "wasm32", derive(Deserialize, Tsify))]
#[cfg_attr(target_arch = "wasm32", serde(rename_all = "camelCase"))]
#[derive(Debug, Default)]
pub struct TransactionOptions {
    /// Whether or not to run a phantom query (i.e., a query that only influences Prisma event logs, but not the database itself)
    /// before opening a transaction, committing, or rollbacking.
    pub use_phantom_query: bool,
}