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
use serde::{ser::SerializeMap, Serialize, Serializer};
use std::cmp::Ordering;
use std::{cell::RefCell, collections::HashMap};

/// Model operations are serialized as an array of objects, each one
/// corresponding to a model. All other operations are simply an object
/// of read and write operation names that can be called on the engine.
#[derive(Debug, Serialize, Default)]
#[serde(rename_all = "camelCase")]
pub struct DmmfOperationMappings {
    pub model_operations: Vec<DmmfModelOperations>,
    pub other_operations: DmmfNonModelOperations,
}

#[derive(Debug, Serialize, Default)]
#[serde(rename_all = "camelCase")]
pub struct DmmfNonModelOperations {
    pub read: Vec<String>,
    pub write: Vec<String>,
}

#[derive(Debug, Default)]
pub struct DmmfModelOperations {
    pub model_name: String,
    pub operations: RefCell<HashMap<String, String>>,
}

impl DmmfModelOperations {
    pub fn new(model_name: String) -> Self {
        Self {
            model_name,
            operations: RefCell::new(HashMap::new()),
        }
    }

    pub fn add_operation(&self, key: String, value: String) {
        self.operations.borrow_mut().insert(key, value);
    }

    pub fn finalize(&self) -> HashMap<String, String> {
        // Cloning required to make the custom serializer work.
        let mut map = self.operations.borrow().clone();

        map.insert("model".into(), self.model_name.clone());
        map
    }
}

/// Serializes a DmmfMapping into a single map.
impl Serialize for DmmfModelOperations {
    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
    where
        S: Serializer,
    {
        let finalized = self.finalize();
        let mut finalized: Vec<(String, String)> = finalized.into_iter().collect();

        // Sugar only: model is the first field in the JSON.
        finalized.sort_unstable_by(|a, b| {
            if a.0 == "model" {
                Ordering::Less
            } else if b.0 == "model" {
                Ordering::Greater
            } else {
                a.0.partial_cmp(&b.0).unwrap()
            }
        });

        let mut map = serializer.serialize_map(Some(finalized.len()))?;
        for (k, v) in finalized {
            map.serialize_entry(&k, &v)?;
        }

        map.end()
    }
}