A typed client for ClickHouse.
serde
for encoding/decoding rows.RowBinary
encoding.To use the crate, add this to your Cargo.toml
:
```toml
[dependencies]
clickhouse =\1"0.11.0"
[dev-dependencies] clickhouse =\1"0.11.0", features = ["test-util"] } ```
See examples.
CH server older than v22.6 (2022-06-16) handles RowBinary
incorrectly in some rare cases. Enable wa-37420
feature to solve this problem. Don't use it for newer versions.
Client
```rust,ignore use clickhouse::Client;
let client = Client::default() .withurl("http://localhost:8123") .withuser("name") .withpassword("123") .withdatabase("test"); ```
```rust,ignore use serde::Deserialize; use clickhouse::Row;
struct MyRow<'a> { no: u32, name: &'a str, }
let mut cursor = client
.query("SELECT ?fields FROM some WHERE no BETWEEN ? AND ?")
.bind(500)
.bind(504)
.fetch::
while let Some(row) = cursor.next().await? { .. } ```
?fields
is replaced with no, name
(fields of Row
).?
is replaced with values in following bind()
calls.fetch_one::<Row>()
and fetch_all::<Row>()
can be used to get a first row or all rows correspondingly.sql::Identifier
can be used to bind table names.Nested
types.rust,ignore
let mut insert = client.insert("some")?;
insert.write(&Row { no: 0, name: "foo" }).await?;
insert.write(&Row { no: 1, name: "bar" }).await?;
insert.end().await?;
end()
isn't called, the INSERT
is aborted.max_insert_block_size
.``rust,ignore
let mut inserter = client.inserter("some")?
.with_max_entries(500_000) //
250_000by default
.with_period(Some(Duration::from_secs(15))); //
None` by default
inserter.write(&Row { no: 0, name: "foo" }).await?; inserter.write(&Row { no: 1, name: "bar" }).await?; let stats = inserter.commit().await?; if stats.entries > 0 { println!( "{} entries ({} transactions) have been inserted", stats.entries, stats.transactions, ); } ```
Inserter
ends an active insert in commit()
if thresholds (max_entries
, period
) are reached.INSERT
s can be biased by using with_period_bias
to avoid load spikes by parallel inserters.commit()
calls are inserted in the same INSERT
statement.rust,ignore
inserter.end().await?;
rust,ignore
client.query("DROP TABLE IF EXISTS some").execute().await?;
Requires the watch
feature.
```rust,ignore
let mut cursor = client
.watch("SELECT max(no), argMax(name, no) FROM some")
.fetch::
let (version, row) = cursor.next().await?.unwrap(); println!("live view updated: version={}, row={:?}", version, row);
// Use only_events()
to iterate over versions only.
let mut cursor = client.watch("someliveview").limit(20).only_events().fetch()?;
println!("live view updated: version={:?}", cursor.next().await?);
```
lv_{sha1(query)}
to reuse the same live view by parallel watchers.JSONEachRowWithProgress
under the hood because of the issue.fetch::<u64>()
and other without specified names.Requires the uuid
feature.
```rust,ignore
struct MyRow { #[serde(with = "clickhouse::uuid")] uuid: uuid::Uuid, } ```
The crate provides utils for mocking CH server and testing DDL, SELECT, INSERT and WATCH queries.
The functionality can be enabled with the test-util
feature. Use it only in dev-dependencies.
See the example.