rust-postgres/tokio-postgres/src/copy_out.rs

57 lines
1.6 KiB
Rust
Raw Normal View History

2019-08-01 03:15:17 +00:00
use crate::client::{InnerClient, Responses};
2019-08-01 04:19:56 +00:00
use crate::codec::FrontendMessage;
use crate::connection::RequestMessages;
use crate::types::ToSql;
use crate::{query, Error, Statement};
2019-08-01 03:15:17 +00:00
use bytes::Bytes;
2019-10-09 02:01:34 +00:00
use futures::{ready, Stream};
2019-08-01 03:15:17 +00:00
use postgres_protocol::message::backend::Message;
use std::pin::Pin;
use std::task::{Context, Poll};
2019-10-09 02:01:34 +00:00
pub async fn copy_out<'a, I>(
client: &InnerClient,
statement: Statement,
params: I,
2019-10-09 02:01:34 +00:00
) -> Result<CopyStream, Error>
where
2019-10-09 02:01:34 +00:00
I: IntoIterator<Item = &'a dyn ToSql>,
I::IntoIter: ExactSizeIterator,
{
let buf = query::encode(client, &statement, params)?;
2019-10-09 02:01:34 +00:00
let responses = start(client, buf).await?;
Ok(CopyStream { responses })
2019-08-01 03:15:17 +00:00
}
async fn start(client: &InnerClient, buf: Bytes) -> Result<Responses, Error> {
2019-08-01 03:15:17 +00:00
let mut responses = client.send(RequestMessages::Single(FrontendMessage::Raw(buf)))?;
match responses.next().await? {
Message::BindComplete => {}
_ => return Err(Error::unexpected_message()),
}
match responses.next().await? {
Message::CopyOutResponse(_) => {}
_ => return Err(Error::unexpected_message()),
}
Ok(responses)
}
2019-10-09 02:01:34 +00:00
pub struct CopyStream {
2019-08-01 03:15:17 +00:00
responses: Responses,
}
2019-10-09 02:01:34 +00:00
impl Stream for CopyStream {
2019-08-01 03:15:17 +00:00
type Item = Result<Bytes, Error>;
fn poll_next(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
match ready!(self.responses.poll_next(cx)?) {
Message::CopyData(body) => Poll::Ready(Some(Ok(body.into_bytes()))),
Message::CopyDone => Poll::Ready(None),
_ => Poll::Ready(Some(Err(Error::unexpected_message()))),
}
}
}