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
94
95
96
97
98
99
100
101
use crate::internals::stream_controller::*;
use crate::prelude::*;
use std::{
  marker::PhantomData,
  sync::{Arc, RwLock},
};

#[derive(Clone)]
pub struct BufferWithCount<Item>
where
  Item: Clone + Send + Sync,
{
  count: usize,
  _item: PhantomData<Item>,
}

impl<'a, Item> BufferWithCount<Item>
where
  Item: Clone + Send + Sync,
{
  pub fn new(count: usize) -> BufferWithCount<Item> {
    assert!(count > 0);
    BufferWithCount { count, _item: PhantomData }
  }
  pub fn execute(
    &self,
    source: Observable<'a, Item>,
  ) -> Observable<'a, Vec<Item>> {
    let count = self.count;
    Observable::<Vec<Item>>::create(move |s| {
      let results = Arc::new(RwLock::new(Vec::new()));
      let sctl = StreamController::new(s);

      let sctl_f = sctl.clone();
      let results_f = Arc::clone(&results);
      let register = move |item: Item| {
        let vec = {
          let mut vec = results_f.write().unwrap();
          vec.push(item);
          if vec.len() == count {
            let v = vec.clone();
            vec.clear();
            Some(v)
          } else {
            None
          }
        };
        if let Some(vec) = vec {
          sctl_f.sink_next(vec);
        }
      };

      {
        let register = register.clone();
        let sctl_error = sctl.clone();
        let sctl_complete = sctl.clone();

        source.inner_subscribe(sctl.new_observer(
          move |_, x| register(x),
          move |_, e| {
            sctl_error.sink_error(e);
          },
          move |serial| {
            let vec = results.read().unwrap();
            if vec.len() > 0 {
              sctl_complete.sink_next(vec.clone());
            }
            sctl_complete.sink_complete(&serial);
          },
        ));
      }
    })
  }
}

impl<'a, Item> Observable<'a, Item>
where
  Item: Clone + Send + Sync,
{
  pub fn buffer_with_count(&self, count: usize) -> Observable<'a, Vec<Item>> {
    BufferWithCount::new(count).execute(self.clone())
  }
}

#[cfg(test)]
mod test {
  use crate::prelude::*;
  use std::{thread, time};

  #[test]
  fn basic() {
    observables::from_iter(0..10)
      .buffer_with_count(3)
      .subscribe(
        print_next_fmt!("{:?}"),
        print_error!(),
        print_complete!(),
      );
    thread::sleep(time::Duration::from_millis(1000));
  }
}