//! An asynchronous, pipelined, PostgreSQL client. //! //! # Example //! //! ```no_run //! use tokio_postgres::{NoTls, Error}; //! //! # #[cfg(not(feature = "runtime"))] fn main() {} //! # #[cfg(feature = "runtime")] //! #[tokio::main] // By default, tokio_postgres uses the tokio crate as its runtime. //! async fn main() -> Result<(), Error> { //! // Connect to the database. //! let (client, connection) = //! tokio_postgres::connect("host=localhost user=postgres", NoTls).await?; //! //! // The connection object performs the actual communication with the database, //! // so spawn it off to run on its own. //! tokio::spawn(async move { //! if let Err(e) = connection.await { //! eprintln!("connection error: {}", e); //! } //! }); //! //! // Now we can prepare a simple statement that just returns its parameter. //! let stmt = client.prepare("SELECT $1::TEXT").await?; //! //! // And then execute it, returning a list of the resulting rows. //! let rows = client //! .query(&stmt, &[&"hello world"]) //! .await?; //! //! // Now we can check that we got back the same string we sent over. //! let value: &str = rows[0].get(0); //! assert_eq!(value, "hello world"); //! //! Ok(()) //! } //! ``` //! //! # Behavior //! //! Calling a method like `Client::query` on its own does nothing. The associated request is not sent to the database //! until the future returned by the method is first polled. Requests are executed in the order that they are first //! polled, not in the order that their futures are created. //! //! # Pipelining //! //! The client supports *pipelined* requests. Pipelining can improve performance in use cases in which multiple, //! independent queries need to be executed. In a traditional workflow, each query is sent to the server after the //! previous query completes. In contrast, pipelining allows the client to send all of the queries to the server up //! front, minimizing time spent by one side waiting for the other to finish sending data: //! //! ```not_rust //! Sequential Pipelined //! | Client | Server | | Client | Server | //! |----------------|-----------------| |----------------|-----------------| //! | send query 1 | | | send query 1 | | //! | | process query 1 | | send query 2 | process query 1 | //! | receive rows 1 | | | send query 3 | process query 2 | //! | send query 2 | | | receive rows 1 | process query 3 | //! | | process query 2 | | receive rows 2 | | //! | receive rows 2 | | | receive rows 3 | | //! | send query 3 | | //! | | process query 3 | //! | receive rows 3 | | //! ``` //! //! In both cases, the PostgreSQL server is executing the queries sequentially - pipelining just allows both sides of //! the connection to work concurrently when possible. //! //! Pipelining happens automatically when futures are polled concurrently (for example, by using the futures `join` //! combinator): //! //! ```rust //! use futures::future; //! use std::future::Future; //! use tokio_postgres::{Client, Error, Statement}; //! //! async fn pipelined_prepare( //! client: &Client, //! ) -> Result<(Statement, Statement), Error> //! { //! future::try_join( //! client.prepare("SELECT * FROM foo"), //! client.prepare("INSERT INTO bar (id, name) VALUES ($1, $2)") //! ).await //! } //! ``` //! //! # Runtime //! //! The client works with arbitrary `AsyncRead + AsyncWrite` streams. Convenience APIs are provided to handle the //! connection process, but these are gated by the `runtime` Cargo feature, which is enabled by default. If disabled, //! all dependence on the tokio runtime is removed. //! //! # SSL/TLS support //! //! TLS support is implemented via external libraries. `Client::connect` and `Config::connect` take a TLS implementation //! as an argument. The `NoTls` type in this crate can be used when TLS is not required. Otherwise, the //! `postgres-openssl` and `postgres-native-tls` crates provide implementations backed by the `openssl` and `native-tls` //! crates, respectively. #![doc(html_root_url = "https://docs.rs/tokio-postgres/0.5")] #![warn(rust_2018_idioms, clippy::all, missing_docs)] pub use crate::client::Client; pub use crate::config::Config; pub use crate::connection::Connection; use crate::error::DbError; pub use crate::error::Error; pub use crate::portal::Portal; pub use crate::query::RowStream; pub use crate::row::{Row, SimpleQueryRow}; pub use crate::simple_query::SimpleQueryStream; #[cfg(feature = "runtime")] pub use crate::socket::Socket; pub use crate::statement::{Column, Statement}; pub use crate::copy_out::CopyStream; #[cfg(feature = "runtime")] use crate::tls::MakeTlsConnect; pub use crate::tls::NoTls; pub use crate::to_statement::ToStatement; pub use crate::transaction::Transaction; use crate::types::ToSql; mod bind; #[cfg(feature = "runtime")] mod cancel_query; mod cancel_query_raw; mod client; mod codec; pub mod config; #[cfg(feature = "runtime")] mod connect; mod connect_raw; #[cfg(feature = "runtime")] mod connect_socket; mod connect_tls; mod connection; mod copy_in; mod copy_out; pub mod error; mod maybe_tls_stream; mod portal; mod prepare; mod query; pub mod row; mod simple_query; #[cfg(feature = "runtime")] mod socket; mod statement; pub mod tls; mod to_statement; mod transaction; pub mod types; /// A convenience function which parses a connection string and connects to the database. /// /// See the documentation for [`Config`] for details on the connection string format. /// /// Requires the `runtime` Cargo feature (enabled by default). /// /// [`Config`]: config/struct.Config.html #[cfg(feature = "runtime")] pub async fn connect( config: &str, tls: T, ) -> Result<(Client, Connection), Error> where T: MakeTlsConnect, { let config = config.parse::()?; config.connect(tls).await } /// An asynchronous notification. #[derive(Clone, Debug)] pub struct Notification { process_id: i32, channel: String, payload: String, } impl Notification { /// The process ID of the notifying backend process. pub fn process_id(&self) -> i32 { self.process_id } /// The name of the channel that the notify has been raised on. pub fn channel(&self) -> &str { &self.channel } /// The "payload" string passed from the notifying process. pub fn payload(&self) -> &str { &self.payload } } /// An asynchronous message from the server. #[allow(clippy::large_enum_variant)] pub enum AsyncMessage { /// A notice. /// /// Notices use the same format as errors, but aren't "errors" per-se. Notice(DbError), /// A notification. /// /// Connections can subscribe to notifications with the `LISTEN` command. Notification(Notification), #[doc(hidden)] __NonExhaustive, } /// Message returned by the `SimpleQuery` stream. pub enum SimpleQueryMessage { /// A row of data. Row(SimpleQueryRow), /// A statement in the query has completed. /// /// The number of rows modified or selected is returned. CommandComplete(u64), #[doc(hidden)] __NonExhaustive, } fn slice_iter<'a>( s: &'a [&'a (dyn ToSql + Sync)], ) -> impl ExactSizeIterator + 'a { s.iter().map(|s| *s as _) }