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
use std::fmt::Debug;
use std::fmt::Display;
use std::fmt;

use crate::core::*;
use crate::store::*;

/// changes that will be applied to store
/// add/update has same effect
#[derive(Debug)]
pub enum LSUpdate<S, C>
where
    S: Spec,
    S::Status: PartialEq,
    C: MetadataItem,
{
    Mod(MetadataStoreObject<S, C>),
    Delete(S::IndexKey),
}

/// changes that happened in the store
#[derive(Debug, PartialEq, Clone)]
pub enum LSChange<S, C>
where
    S: Spec,
    S::Status: PartialEq,
    C: MetadataItem,
{
    Add(MetadataStoreObject<S, C>),
    Mod(MetadataStoreObject<S, C>, MetadataStoreObject<S, C>), // new, old
    Delete(MetadataStoreObject<S, C>),
}

impl<S, C> fmt::Display for LSChange<S, C>
where
    S: Spec,
    S::IndexKey: Display,
    S::Status: PartialEq,
    C: MetadataItem,
{
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        match self {
            Self::Add(add) => write!(f, "{} Add: {}", S::LABEL, add.key()),
            Self::Mod(update, _) => write!(f, "{} Mod: {}", S::LABEL, update.key()),
            Self::Delete(del) => write!(f, "{} Delete: {}", S::LABEL, del.key()),
        }
    }
}

impl<S, C> LSChange<S, C>
where
    S: Spec,
    S::Status: PartialEq,
    C: MetadataItem,
{
    pub fn add(value: impl Into<MetadataStoreObject<S, C>>) -> Self {
        LSChange::Add(value.into())
    }

    pub fn update(new: MetadataStoreObject<S, C>, old: MetadataStoreObject<S, C>) -> Self {
        LSChange::Mod(new, old)
    }

    pub fn delete(value: MetadataStoreObject<S, C>) -> Self {
        LSChange::Delete(value)
    }
}

/// change in spec
#[derive(Debug, Eq, PartialEq, Clone)]
pub enum SpecChange<S>
where
    S: Spec,
{
    Add(S),
    Mod(S, S),
    Delete(S),
}

impl<S, C> From<LSChange<S, C>> for SpecChange<S>
where
    S: Spec,
    S::Status: PartialEq,
    C: MetadataItem,
{
    fn from(meta: LSChange<S, C>) -> Self {
        match meta {
            LSChange::Add(val) => Self::Add(val.spec),
            LSChange::Mod(new, old) => Self::Mod(new.spec, old.spec),
            LSChange::Delete(old) => Self::Delete(old.spec),
        }
    }
}