rx_rust/operators/mathematical_aggregate/
sum.rs1use crate::utils::types::NecessarySend;
2use crate::{
3 disposable::subscription::Subscription,
4 observable::Observable,
5 observer::{Observer, Termination},
6};
7use educe::Educe;
8use std::ops::AddAssign;
9
10#[derive(Educe)]
37#[educe(Debug, Clone)]
38pub struct Sum<OE> {
39 source: OE,
40}
41
42impl<OE> Sum<OE> {
43 pub fn new<'or, 'sub, T, E>(source: OE) -> Self
44 where
45 OE: Observable<'or, 'sub, T, E>,
46 {
47 Self { source }
48 }
49}
50
51impl<'or, 'sub, T, E, OE> Observable<'or, 'sub, T, E> for Sum<OE>
52where
53 T: AddAssign + NecessarySend + 'or,
54 OE: Observable<'or, 'sub, T, E>,
55{
56 fn subscribe(self, observer: impl Observer<T, E> + NecessarySend + 'or) -> Subscription<'sub> {
57 let observer = SumObserver {
58 observer,
59 sum: None,
60 };
61 self.source.subscribe(observer)
62 }
63}
64
65struct SumObserver<T, OR> {
66 observer: OR,
67 sum: Option<T>,
68}
69
70impl<T, E, OR> Observer<T, E> for SumObserver<T, OR>
71where
72 T: AddAssign,
73 OR: Observer<T, E>,
74{
75 fn on_next(&mut self, value: T) {
76 if let Some(sum) = &mut self.sum {
77 *sum += value;
78 } else {
79 self.sum = Some(value);
80 }
81 }
82
83 fn on_termination(mut self, termination: Termination<E>) {
84 match termination {
85 Termination::Completed => {
86 if let Some(sum) = self.sum {
87 self.observer.on_next(sum);
88 }
89 }
90 Termination::Error(_) => {}
91 }
92 self.observer.on_termination(termination)
93 }
94}