use serde_json;
use futures::{Future, IntoFuture, Poll};
use futures_cpupool::CpuPool;
use serde::Serialize;
use error::{self, Error, Result};
use client::{AsyncSender, Client, Sender, SyncSender};
use client::requests::RequestBuilder;
use client::requests::params::{Id, Index, Type};
use client::requests::endpoints::IndexRequest;
use client::requests::raw::RawRequestInner;
use client::responses::IndexResponse;
use types::document::DocumentType;
pub type IndexRequestBuilder<TSender, TDocument> = RequestBuilder<TSender, IndexRequestInner<TDocument>>;
#[doc(hidden)]
pub struct IndexRequestInner<TDocument> {
index: Index<'static>,
ty: Type<'static>,
id: Id<'static>,
doc: TDocument,
}
impl<TSender> Client<TSender>
where
TSender: Sender,
{
pub fn document_index<TDocument>(&self, index: Index<'static>, id: Id<'static>, doc: TDocument) -> IndexRequestBuilder<TSender, TDocument>
where
TDocument: Serialize + DocumentType,
{
let ty = TDocument::name().into();
RequestBuilder::new(
self.clone(),
None,
IndexRequestInner {
index: index,
ty: ty,
id: id,
doc: doc,
},
)
}
}
impl<TDocument> IndexRequestInner<TDocument>
where
TDocument: Serialize,
{
fn into_sync_request(self) -> Result<IndexRequest<'static, Vec<u8>>> {
let body = serde_json::to_vec(&self.doc).map_err(error::request)?;
Ok(IndexRequest::for_index_ty_id(
self.index,
self.ty,
self.id,
body,
))
}
}
impl<TDocument> IndexRequestInner<TDocument>
where
TDocument: Serialize + Send + 'static,
{
fn into_async_request(self, ser_pool: Option<CpuPool>) -> Box<Future<Item = IndexRequest<'static, Vec<u8>>, Error = Error>> {
if let Some(ser_pool) = ser_pool {
let request_future = ser_pool.spawn_fn(|| self.into_sync_request());
Box::new(request_future)
} else {
Box::new(self.into_sync_request().into_future())
}
}
}
impl<TSender, TDocument> IndexRequestBuilder<TSender, TDocument>
where
TSender: Sender,
{
pub fn ty<I>(mut self, ty: I) -> Self
where
I: Into<Type<'static>>,
{
self.inner.ty = ty.into();
self
}
}
impl<TDocument> IndexRequestBuilder<SyncSender, TDocument>
where
TDocument: Serialize,
{
pub fn send(self) -> Result<IndexResponse> {
let req = self.inner.into_sync_request()?;
RequestBuilder::new(self.client, self.params, RawRequestInner::new(req))
.send()?
.into_response()
}
}
impl<TDocument> IndexRequestBuilder<AsyncSender, TDocument>
where
TDocument: Serialize + Send + 'static,
{
pub fn send(self) -> Pending {
let (client, params) = (self.client, self.params);
let ser_pool = client.sender.serde_pool.clone();
let req_future = self.inner.into_async_request(ser_pool);
let res_future = req_future.and_then(move |req| {
RequestBuilder::new(client, params, RawRequestInner::new(req))
.send()
.and_then(|res| res.into_response())
});
Pending::new(res_future)
}
}
pub struct Pending {
inner: Box<Future<Item = IndexResponse, Error = Error>>,
}
impl Pending {
fn new<F>(fut: F) -> Self
where
F: Future<Item = IndexResponse, Error = Error> + 'static,
{
Pending {
inner: Box::new(fut),
}
}
}
impl Future for Pending {
type Item = IndexResponse;
type Error = Error;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
self.inner.poll()
}
}
#[cfg(test)]
mod tests {
use serde_json::Value;
use prelude::*;
#[test]
fn default_request() {
let client = SyncClientBuilder::new().build().unwrap();
let req = client
.document_index(index("test-idx"), id("1"), Value::Null)
.inner
.into_sync_request()
.unwrap();
assert_eq!("/test-idx/value/1", req.url.as_ref());
assert_eq!("null".as_bytes().to_vec(), req.body);
}
#[test]
fn specify_ty() {
let client = SyncClientBuilder::new().build().unwrap();
let req = client
.document_index(index("test-idx"), id("1"), Value::Null)
.ty("new-ty")
.inner
.into_sync_request()
.unwrap();
assert_eq!("/test-idx/new-ty/1", req.url.as_ref());
}
#[test]
fn document_borrow() {
let client = SyncClientBuilder::new().build().unwrap();
let doc = Value::Null;
let req = client
.document_index(index("test-idx"), id("1"), &doc)
.inner
.into_sync_request()
.unwrap();
assert_eq!("/test-idx/value/1", req.url.as_ref());
}
}