8 releases (4 breaking)

0.4.2 May 9, 2024
0.4.1 Jun 22, 2023
0.4.0 May 15, 2023
0.3.0 Jul 26, 2022
0.0.0 Oct 27, 2020

#1368 in Database interfaces

Download history 105/week @ 2024-08-18 224/week @ 2024-08-25 115/week @ 2024-09-01 69/week @ 2024-09-08 93/week @ 2024-09-15 189/week @ 2024-09-22 144/week @ 2024-09-29 158/week @ 2024-10-06 154/week @ 2024-10-13 71/week @ 2024-10-20 63/week @ 2024-10-27 140/week @ 2024-11-03 127/week @ 2024-11-10 116/week @ 2024-11-17 110/week @ 2024-11-24 68/week @ 2024-12-01

430 downloads per month
Used in 11 crates (7 directly)

MIT/Apache

32KB
688 lines

EdgeDB Rust Binding: Errors Crate

This crate contains definitions of errors returned from the database.

License

Licensed under either of

at your option.


lib.rs:

Error Handling for EdgeDB

All errors that EdgeDB Rust bindings produce are encapsulated into the Error structure. The structure is a bit like Box<dyn Error> or anyhow::Error, except it can only contain EdgeDB error types. Or UserError can be used to encapsulate custom errors (commonly used to return an error from a transaction).

A full list of EdgeDB error types on a single page can be found on the website documentation.

Each error kind is represented as a separate type that implements the ErrorKind trait. But error kinds are used like marker structs; you can use Error::is for error kinds and use them to create instances of the error:

let err = UserError::with_source(io::Error::from(io::ErrorKind::NotFound));
assert!(err.is::<UserError>());

Since errors are hirarchical, Error::is works with any ancestor:

assert!(err.is::<MissingArgumentError>());
assert!(err.is::<QueryArgumentError>());  // implied by the assertion above
assert!(err.is::<InterfaceError>());  // and this one
assert!(err.is::<ClientError>());  // and this one

Error hierarchy doesn't have multiple inheritance (i.e. every error has only single parent). When we match across different parents we use error tags:


assert!(err1.is::<ClientConnectionTimeoutError>());
assert!(err2.is::<TransactionConflictError>());
// Both of these are retried
assert!(err1.has_tag(SHOULD_RETRY));
assert!(err2.has_tag(SHOULD_RETRY));

// But they aren't a part of common hierarchy
assert!(err1.is::<ClientError>());
assert!(!err1.is::<ExecutionError>());
assert!(err2.is::<ExecutionError>());
assert!(!err2.is::<ClientError>());

Errors in Transactions

Special care for errors must be taken in transactions. Generally:

  1. Errors from queries should not be ignored, and should be propagagated up to the transaction function.
  2. User errors can be encapsulated into UserError via one of the methods:
  1. Original query error must be propagated via error chain. It can be in the .source() chain but must not be swallowed, otherwise retrying transaction may work incorrectly.

Nice Error Reporting

Refer to documentation in the edgedb-tokio crate.

Dependencies

~135–550KB