1
use futures::{Future, FutureExt};
2

            
3
use super::{BuilderState, Command, KeyOperation, KeyValue, Output};
4
use crate::{
5
    keyvalue::{IncompatibleTypeError, Numeric, Value},
6
    Error,
7
};
8

            
9
/// Executes [`Command::Set`] when awaited. Also offers methods to customize the
10
/// options for the operation.
11
#[must_use = "futures do nothing unless you `.await` or poll them"]
12
pub struct Builder<'a, KeyValue, V> {
13
    state: BuilderState<'a, Options<'a, KeyValue>, Result<V, Error>>,
14
}
15

            
16
struct Options<'a, KeyValue> {
17
    kv: &'a KeyValue,
18
    namespace: Option<String>,
19
    key: String,
20
    increment: bool,
21
    amount: Numeric,
22
    saturating: bool,
23
}
24

            
25
impl<'a, K, V> Builder<'a, K, V>
26
where
27
    K: KeyValue,
28
{
29
40089
    pub(crate) fn new(
30
40089
        kv: &'a K,
31
40089
        namespace: Option<String>,
32
40089
        increment: bool,
33
40089
        key: String,
34
40089
        amount: Numeric,
35
40089
    ) -> Self {
36
40089
        Self {
37
40089
            state: BuilderState::Pending(Some(Options {
38
40089
                key,
39
40089
                kv,
40
40089
                namespace,
41
40089
                increment,
42
40089
                amount,
43
40089
                saturating: true,
44
40089
            })),
45
40089
        }
46
40089
    }
47

            
48
20
    fn options(&mut self) -> &mut Options<'a, K> {
49
20
        if let BuilderState::Pending(Some(options)) = &mut self.state {
50
20
            options
51
        } else {
52
            panic!("Attempted to use after retrieving the result")
53
        }
54
20
    }
55

            
56
    /// Allows overflowing the value.
57
20
    pub fn allow_overflow(mut self) -> Self {
58
20
        self.options().saturating = false;
59
20
        self
60
20
    }
61
}
62

            
63
impl<'a, K, V> Future for Builder<'a, K, V>
64
where
65
    K: KeyValue,
66
    V: TryFrom<Numeric, Error = IncompatibleTypeError>,
67
{
68
    type Output = Result<V, Error>;
69

            
70
100297
    fn poll(
71
100297
        mut self: std::pin::Pin<&mut Self>,
72
100297
        cx: &mut std::task::Context<'_>,
73
100297
    ) -> std::task::Poll<Self::Output> {
74
100297
        match &mut self.state {
75
60208
            BuilderState::Executing(future) => future.as_mut().poll(cx),
76
40089
            BuilderState::Pending(builder) => {
77
40089
                let Options {
78
40089
                    kv,
79
40089
                    namespace,
80
40089
                    key,
81
40089
                    increment,
82
40089
                    amount,
83
40089
                    saturating,
84
40089
                } = builder.take().expect("expected builder to have options");
85
40089
                let future = async move {
86
40077
                    let result = kv
87
                        .execute_key_operation(KeyOperation {
88
40089
                            namespace,
89
40089
                            key,
90
40089
                            command: if increment {
91
40049
                                Command::Increment { amount, saturating }
92
                            } else {
93
40
                                Command::Decrement { amount, saturating }
94
                            },
95
20119
                        })
96
20119
                        .await?;
97
40077
                    if let Output::Value(Some(Value::Numeric(value))) = result {
98
40077
                        Ok(V::try_from(value).expect("server should send back identical type"))
99
                    } else {
100
                        unreachable!("Unexpected result from key value operation")
101
                    }
102
40089
                }
103
40089
                .boxed();
104
40089

            
105
40089
                self.state = BuilderState::Executing(future);
106
40089
                self.poll(cx)
107
            }
108
        }
109
100297
    }
110
}