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
use context::{Context};
use std::vec::Vec;
use std::cell::Cell;
use std::boxed::Box;
use futures::{future, Future};
use tokio::reactor::Handle;
use std::io;

// pub type Middleware<T: Context> = fn(T, chain: &MiddlewareChain<T>) -> T;
pub type MiddlewareReturnValue<T> = Box<Future<Item=T, Error=io::Error> + Send>;
pub type Middleware<T> = fn(T, chain: &MiddlewareChain<T>) -> MiddlewareReturnValue<T>;

pub struct MiddlewareChain<'a, T: Context> {
  _chain_index: Cell<usize>,
  pub handle: &'a Handle,
  pub middleware: Vec<Middleware<T>>
}

impl<'a, T: 'static + Context + Send> MiddlewareChain<'a, T> {
  pub fn new(middleware: Vec<Middleware<T>>, handle: &'a Handle) -> MiddlewareChain<'a, T> {
    MiddlewareChain {
      handle: handle,
      middleware: middleware,
      _chain_index: Cell::new(0),
    }
  }

  pub fn next(&self, context: T) -> MiddlewareReturnValue<T> {
    let next_middleware = self.middleware.get(self._chain_index.get());
    self._chain_index.set(self._chain_index.get() + 1);

    match next_middleware {
      Some(middleware) => middleware(context, self),
      None => Box::new(future::ok(context))
    }
  }
}