2013-09-03 06:19:03 +00:00
|
|
|
Rust-Postgres
|
2013-08-18 21:16:39 +00:00
|
|
|
=============
|
2013-09-03 06:19:03 +00:00
|
|
|
A native PostgreSQL driver for Rust.
|
|
|
|
|
2013-10-17 03:37:51 +00:00
|
|
|
Documentation is available at http://sfackler.com/doc/rust-postgres.
|
2013-09-30 04:21:00 +00:00
|
|
|
|
2013-10-08 06:03:05 +00:00
|
|
|
[![Build Status](https://travis-ci.org/sfackler/rust-postgres.png?branch=master)](https://travis-ci.org/sfackler/rust-postgres)
|
|
|
|
|
2013-09-03 06:19:03 +00:00
|
|
|
Overview
|
|
|
|
========
|
|
|
|
Rust-Postgres is a pure-Rust frontend for the popular PostgreSQL database. It
|
|
|
|
exposes a high level interface in the vein of JDBC or Go's `database/sql`
|
|
|
|
package.
|
|
|
|
```rust
|
2013-10-16 00:37:34 +00:00
|
|
|
extern mod postgres = "github.com/sfackler/rust-postgres";
|
2013-10-01 07:01:54 +00:00
|
|
|
extern mod extra;
|
2013-09-03 06:19:03 +00:00
|
|
|
|
2013-10-01 07:01:54 +00:00
|
|
|
use extra::time;
|
|
|
|
use extra::time::Timespec;
|
|
|
|
|
|
|
|
use postgres::{PostgresConnection, PostgresStatement};
|
2013-09-03 06:19:03 +00:00
|
|
|
use postgres::types::ToSql;
|
|
|
|
|
|
|
|
struct Person {
|
|
|
|
id: i32,
|
|
|
|
name: ~str,
|
2013-10-01 07:01:54 +00:00
|
|
|
time_created: Timespec,
|
2013-09-03 06:19:03 +00:00
|
|
|
data: Option<~[u8]>
|
|
|
|
}
|
|
|
|
|
|
|
|
fn main() {
|
2013-09-08 21:26:34 +00:00
|
|
|
let conn = PostgresConnection::connect("postgres://postgres@localhost");
|
2013-09-03 06:19:03 +00:00
|
|
|
|
|
|
|
conn.update("CREATE TABLE person (
|
2013-10-01 07:01:54 +00:00
|
|
|
id SERIAL PRIMARY KEY,
|
|
|
|
name VARCHAR NOT NULL,
|
|
|
|
time_created TIMESTAMP NOT NULL,
|
|
|
|
data BYTEA
|
2013-09-03 06:19:03 +00:00
|
|
|
)", []);
|
|
|
|
let me = Person {
|
|
|
|
id: 0,
|
|
|
|
name: ~"Steven",
|
2013-10-01 07:01:54 +00:00
|
|
|
time_created: time::get_time(),
|
2013-09-03 06:19:03 +00:00
|
|
|
data: None
|
|
|
|
};
|
2013-10-01 07:01:54 +00:00
|
|
|
conn.update("INSERT INTO person (name, time_created, data)
|
2013-09-03 06:19:03 +00:00
|
|
|
VALUES ($1, $2, $3)",
|
2013-10-01 07:01:54 +00:00
|
|
|
[&me.name as &ToSql, &me.time_created as &ToSql,
|
2013-09-03 06:19:03 +00:00
|
|
|
&me.data as &ToSql]);
|
|
|
|
|
2013-10-01 07:01:54 +00:00
|
|
|
let stmt = conn.prepare("SELECT id, name, time_created, data FROM person");
|
2013-09-03 06:19:03 +00:00
|
|
|
for row in stmt.query([]) {
|
|
|
|
let person = Person {
|
|
|
|
id: row[0],
|
|
|
|
name: row[1],
|
2013-10-01 07:01:54 +00:00
|
|
|
time_created: row[2],
|
2013-09-03 06:19:03 +00:00
|
|
|
data: row[3]
|
|
|
|
};
|
2013-10-01 07:01:54 +00:00
|
|
|
println!("Found person {}", person.name);
|
2013-09-03 06:19:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
Requirements
|
|
|
|
============
|
|
|
|
|
|
|
|
* **Rust** - Rust-Postgres is developed against the *master* branch of the Rust
|
|
|
|
repository. It will most likely not build against the releases on
|
|
|
|
http://www.rust-lang.org.
|
|
|
|
|
|
|
|
* **PostgreSQL 7.4 or later** - Rust-Postgres speaks version 3 of the
|
|
|
|
PostgreSQL protocol, which corresponds to versions 7.4 and later. If your
|
|
|
|
version of Postgres was compiled in the last decade, you should be okay.
|
|
|
|
|
|
|
|
Usage
|
|
|
|
=====
|
|
|
|
|
|
|
|
Connecting
|
|
|
|
----------
|
|
|
|
Connect to a Postgres server using the standard URI format:
|
|
|
|
```rust
|
|
|
|
let conn = PostgresConnection::connect("postgres://user:pass@host:port/database?arg1=val1&arg2=val2");
|
|
|
|
```
|
|
|
|
`pass` may be omitted if not needed. `port` defaults to `5432` and `database`
|
2013-10-28 15:42:57 +00:00
|
|
|
defaults to the value of `user` if not specified. The driver supports `trust`
|
|
|
|
and `password` authentication.
|
2013-09-03 06:19:03 +00:00
|
|
|
|
|
|
|
Statement Preparation
|
|
|
|
---------------------
|
|
|
|
Prepared statements can have parameters, represented as `$n` where `n` is an
|
|
|
|
index into the parameter array starting from 1:
|
|
|
|
```rust
|
|
|
|
let stmt = conn.prepare("SELECT * FROM foo WHERE bar = $1 AND baz = $2");
|
|
|
|
```
|
|
|
|
|
|
|
|
Querying
|
|
|
|
--------
|
|
|
|
A prepared statement can be executed with the `query` and `update` methods.
|
|
|
|
Both methods take an array of parameters to bind to the query represented as
|
|
|
|
`&ToSql` trait objects. `update` returns the number of rows affected by the
|
|
|
|
query (or 0 if not applicable):
|
2013-09-03 06:23:58 +00:00
|
|
|
```rust
|
2013-09-03 06:19:03 +00:00
|
|
|
let stmt = conn.prepare("UPDATE foo SET bar = $1 WHERE baz = $2");
|
|
|
|
let updates = stmt.update([&1i32 as &ToSql, & &"biz" as &ToSql]);
|
|
|
|
println!("{} rows were updated", updates);
|
|
|
|
```
|
|
|
|
`query` returns a result iterator. Fields of each row in the result can be
|
2013-09-16 03:05:32 +00:00
|
|
|
accessed either by their indicies or their column names. Unlike statement
|
|
|
|
parameters, result columns are zero-indexed.
|
2013-09-03 06:19:03 +00:00
|
|
|
```rust
|
|
|
|
let stmt = conn.prepare("SELECT bar, baz FROM foo");
|
|
|
|
for row in stmt.query([]) {
|
|
|
|
let bar: i32 = row[0];
|
|
|
|
let baz: ~str = row["baz"];
|
|
|
|
println!("bar: {}, baz: {}", bar, baz);
|
|
|
|
}
|
|
|
|
```
|
|
|
|
In addition, `PostgresConnection` has a utility `update` method which is useful
|
|
|
|
if a statement is only going to be executed once:
|
|
|
|
```rust
|
|
|
|
let updates = conn.update("UPDATE foo SET bar = $1 WHERE baz = $2",
|
|
|
|
[&1i32 as &ToSql, & &"biz" as &ToSql]);
|
|
|
|
println!("{} rows were updated", updates);
|
|
|
|
```
|
|
|
|
|
|
|
|
Transactions
|
|
|
|
------------
|
2013-10-14 01:58:31 +00:00
|
|
|
The `transaction` method will start a new transaction. It returns a
|
|
|
|
`PostgresTransaction` object which has the functionality of a
|
|
|
|
`PostgresConnection` as well as methods to control the result of the
|
|
|
|
transaction:
|
2013-09-03 06:19:03 +00:00
|
|
|
```rust
|
2013-10-14 01:58:31 +00:00
|
|
|
{
|
|
|
|
let trans = conn.transaction();
|
2013-09-03 06:19:03 +00:00
|
|
|
trans.update(...);
|
|
|
|
let stmt = trans.prepare(...);
|
|
|
|
|
|
|
|
if a_bad_thing_happened {
|
|
|
|
trans.set_rollback();
|
|
|
|
}
|
|
|
|
|
|
|
|
if the_coast_is_clear {
|
|
|
|
trans.set_commit();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
2013-10-14 01:58:31 +00:00
|
|
|
The transaction will be active until the `PostgresTransaction` object falls out
|
|
|
|
of scope. A transaction will commit by default. Nested transactions are
|
|
|
|
supported via savepoints.
|
2013-09-03 06:19:03 +00:00
|
|
|
|
2013-09-04 05:55:01 +00:00
|
|
|
Lazy Queries
|
|
|
|
------------
|
2013-09-16 03:05:32 +00:00
|
|
|
Some queries may return a large amount of data. Statements prepared within a
|
|
|
|
transaction have an additional method, `lazy_query`. The rows returned from a
|
|
|
|
call to `lazy_query` are pulled from the database in batches as needed:
|
2013-09-04 05:55:01 +00:00
|
|
|
```rust
|
2013-10-14 01:58:31 +00:00
|
|
|
{
|
|
|
|
let trans = conn.transaction();
|
2013-09-04 05:55:01 +00:00
|
|
|
let stmt = trans.prepare(query)
|
|
|
|
|
|
|
|
// No more than 100 rows will be stored in memory at any time
|
|
|
|
for row in stmt.lazy_query(100, params) {
|
|
|
|
// do things
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
2013-09-03 06:19:03 +00:00
|
|
|
Error Handling
|
|
|
|
--------------
|
|
|
|
The methods described above will fail if there is an error. For each of these
|
|
|
|
methods, there is a second variant prefixed with `try_` which returns a
|
|
|
|
`Result`:
|
|
|
|
```rust
|
2013-09-04 02:35:28 +00:00
|
|
|
match conn.try_update(query, params) {
|
2013-09-03 06:19:03 +00:00
|
|
|
Ok(updates) => println!("{} rows were updated", updates),
|
2013-09-16 02:26:56 +00:00
|
|
|
Err(err) => match err.code {
|
|
|
|
NotNullViolation => println!("Something was NULL that shouldn't be"),
|
|
|
|
SyntaxError => println!("Invalid query syntax"),
|
|
|
|
_ => println!("A bad thing happened: {}", err.message),
|
|
|
|
}
|
2013-09-03 06:19:03 +00:00
|
|
|
}
|
|
|
|
```
|
|
|
|
|
2013-09-18 06:08:42 +00:00
|
|
|
Connection Pooling
|
|
|
|
------------------
|
|
|
|
A very basic fixed-size connection pool is provided in the `pool` module. A
|
|
|
|
single pool can be shared across tasks and `get_connection` will block until a
|
|
|
|
connection is available.
|
|
|
|
```rust
|
2013-09-30 05:03:11 +00:00
|
|
|
let pool = PostgresConnectionPool::new("postgres://postgres@localhost", 5);
|
2013-09-18 06:08:42 +00:00
|
|
|
|
|
|
|
for _ in range(0, 10) {
|
|
|
|
do task::spawn_with(pool.clone()) |pool| {
|
|
|
|
let conn = pool.get_connection();
|
|
|
|
conn.query(...);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
2013-09-03 06:19:03 +00:00
|
|
|
Type Correspondence
|
|
|
|
-------------------
|
|
|
|
Rust-Postgres enforces a strict correspondence between Rust types and Postgres
|
2013-09-16 03:05:32 +00:00
|
|
|
types. The driver currently supports the following conversions:
|
2013-09-03 06:19:03 +00:00
|
|
|
|
|
|
|
<table>
|
|
|
|
<thead>
|
|
|
|
<tr>
|
|
|
|
<td>Rust Type</td>
|
|
|
|
<td>Postgres Type</td>
|
|
|
|
</tr>
|
|
|
|
</thead>
|
|
|
|
<tbody>
|
|
|
|
<tr>
|
|
|
|
<td>bool</td>
|
|
|
|
<td>BOOL</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>i8</td>
|
|
|
|
<td>"char"</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>i16</td>
|
|
|
|
<td>SMALLINT</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>i32</td>
|
|
|
|
<td>INT</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>i64</td>
|
|
|
|
<td>BIGINT</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>f32</td>
|
|
|
|
<td>FLOAT4</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>f64</td>
|
|
|
|
<td>FLOAT8</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>str</td>
|
|
|
|
<td>VARCHAR, CHAR(n), TEXT</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>[u8]</td>
|
|
|
|
<td>BYTEA</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>extra::json::Json</td>
|
|
|
|
<td>JSON</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>extra::uuid::Uuid</td>
|
|
|
|
<td>UUID</td>
|
|
|
|
</tr>
|
2013-09-08 20:27:15 +00:00
|
|
|
<tr>
|
|
|
|
<td>extra::time::Timespec</td>
|
2013-09-09 04:42:03 +00:00
|
|
|
<td>TIMESTAMP, TIMESTAMP WITH TIME ZONE</td>
|
2013-09-08 20:27:15 +00:00
|
|
|
</tr>
|
2013-10-31 05:51:18 +00:00
|
|
|
<tr>
|
|
|
|
<td>types::range::Range<i32></td>
|
|
|
|
<td>INT4RANGE</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>types::range::Range<i64></td>
|
|
|
|
<td>INT8RANGE</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>types::range::Range<Timespec></td>
|
|
|
|
<td>TSRANGE, TSTZRANGE</td>
|
|
|
|
</tr>
|
2013-09-03 06:19:03 +00:00
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
|
|
|
|
More conversions can be defined by implementing the `ToSql` and `FromSql`
|
|
|
|
traits.
|
|
|
|
|
|
|
|
Development
|
|
|
|
===========
|
2013-09-16 03:05:32 +00:00
|
|
|
Rust-Postgres is still in the early stages of development, so don't be
|
|
|
|
surprised if APIs change and things break. If something's not working properly,
|
|
|
|
file an issue or submit a pull request!
|