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
use super::{PrismaValue, TypeIdentifier};
use crate::DomainError;
use bigdecimal::ToPrimitive;

pub(crate) trait PrismaValueExtensions {
    fn coerce(self, to_type: TypeIdentifier) -> crate::Result<PrismaValue>;
}

impl PrismaValueExtensions for PrismaValue {
    // Todo this is not exhaustive for now.
    fn coerce(self, to_type: TypeIdentifier) -> crate::Result<PrismaValue> {
        let coerced = match (self, to_type) {
            // Trivial cases
            (PrismaValue::Null, _) => PrismaValue::Null,
            (val @ PrismaValue::String(_), TypeIdentifier::String) => val,
            (val @ PrismaValue::Int(_), TypeIdentifier::Int) => val,
            (val @ PrismaValue::Float(_), TypeIdentifier::Float) => val,
            (val @ PrismaValue::Float(_), TypeIdentifier::Decimal) => val,
            (val @ PrismaValue::Boolean(_), TypeIdentifier::Boolean) => val,
            (val @ PrismaValue::DateTime(_), TypeIdentifier::DateTime) => val,
            (val @ PrismaValue::Enum(_), TypeIdentifier::Enum(_)) => val,
            (val @ PrismaValue::Uuid(_), TypeIdentifier::UUID) => val,
            (val @ PrismaValue::BigInt(_), TypeIdentifier::BigInt) => val,
            (val @ PrismaValue::Bytes(_), TypeIdentifier::Bytes) => val,
            (val @ PrismaValue::Json(_), TypeIdentifier::Json) => val,

            // Valid String coercions
            (PrismaValue::Int(i), TypeIdentifier::String) => PrismaValue::String(format!("{i}")),
            (PrismaValue::Float(f), TypeIdentifier::String) => PrismaValue::String(f.to_string()),
            (PrismaValue::Boolean(b), TypeIdentifier::String) => PrismaValue::String(format!("{b}")),
            (PrismaValue::Enum(e), TypeIdentifier::String) => PrismaValue::String(e),
            (PrismaValue::Uuid(u), TypeIdentifier::String) => PrismaValue::String(u.to_string()),

            // Valid Int coersions
            (PrismaValue::String(s), TypeIdentifier::Int) => match s.parse() {
                Ok(i) => PrismaValue::Int(i),
                Err(_) => return Err(DomainError::ConversionFailure(format!("{s:?}"), format!("{to_type:?}"))),
            },
            (PrismaValue::Float(f), TypeIdentifier::Int) => PrismaValue::Int(f.to_i64().unwrap()),
            (PrismaValue::BigInt(i), TypeIdentifier::Int) => PrismaValue::Int(i),

            // Valid BigInt coersions
            (PrismaValue::Int(i), TypeIdentifier::BigInt) => PrismaValue::BigInt(i),

            // Todo other coercions here

            // Lists
            (PrismaValue::List(list), typ) => PrismaValue::List(
                list.into_iter()
                    .map(|val| val.coerce(typ))
                    .collect::<crate::Result<Vec<_>>>()?,
            ),

            // Invalid coercion
            (val, typ) => return Err(DomainError::ConversionFailure(format!("{val:?}"), format!("{typ:?}"))),
        };

        Ok(coerced)
    }
}