7 releases (breaking)
0.6.0 | Jun 30, 2022 |
---|---|
0.5.0 | Jun 10, 2022 |
0.4.0 | May 11, 2022 |
0.3.0 | Sep 21, 2021 |
0.1.1 | Aug 16, 2021 |
#1048 in Database interfaces
72 downloads per month
55KB
1K
SLoC
Requeuest
Requeuest (pronounced "recused") is a message queue which acts as an intermediary for HTTP requests, making sure that the sent request gets successfully delivered eventually, meaning that you do not have to implement retry logic for HTTP API requests. The queue uses the sqlxmq
crate to make postgres its store for messages, which avoids the reliability risk of a dedicated message queue service potentially being down. This comes with the trade-off that job runners become part of the library consumer's process, and that a handle to the runner has to be kept alive so jobs can run in the background, since jobs cannot be delegated to a separate runner service.
Getting started
Assuming you already have an sqlx
connection to a postgres database, you will first need to run migrations so the needed tables and SQL functions can get set up on your postgres database. It's recommended to have a distinct database specifically for requeuest, to avoid interference with the migration management of a different service or library.
requeuest::migrate(&pool).await?;
Once that's taken care of, start by constructing a client. This is what you will use to spawn requests, an what will execute jobs in the background. It will keep doing so until it is dropped. The client contains a tokio JoinHandle
which you can remove from the client with the Client::take_listener
method if you want the listener to keep running after the client has dropped, or otherwise interface with the background task directly.
use requeuest::Client;
let client = Client::new(pool, &["my_service"]).await?;
After the client has been constructed, you can begin spawning jobs. Here we send a get request to an example address:
use requeuest::{HeaderMap, Request};
let request = Request::get("https://foo.bar/_api/baz".parse()?, HeaderMap::new());
client.spawn("my_service", &request).await?;
You can also also get the response back from a successfully delivered request.
// You can skip the HeaderMap import by invoking the constructor via the Default trait
let request = Request::post("https://example.com/_api/bar/foo".parse()?, Vec::from("some data"), Default::default());
let response = client.spawn_returning("my_service", &request).await?;
Note that the spawn_returning
method will wait indefinitely (or to be precise, roughly 10^293 years) until a successful response is received by default, so this will wait forever if a request is sent to e.g. an unregistered domain, or a request to an API which that's guaranteed to always get a response back with a non-200 response code.
Testing
requeuest's integration tests rely on sqlx-database-tester
, which expects a postgres database to be available with a user that has permissions to create, edit and drop databases. Which database(s) to use is set with a postgres url (e.g. postgres://postgres:password@localhost/
) in the DATABASE_URL
environment variable, or with an entry in a .env
. See sqlx-database-tester
's documentation for more information.
Pre-commit usage
- If not installed, install with your package manager, or
pip install --user pre-commit
- Run
pre-commit autoupdate
to update the pre-commit config to use the newest template - Run
pre-commit install
to install the pre-commit hooks to your local environment
Dependencies
~14–31MB
~515K SLoC