Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/w-sodalite/reqwest-lb
The reqwest load balancer middleware
https://github.com/w-sodalite/reqwest-lb
load-balancer reqwest reqwest-middleware
Last synced: 29 days ago
JSON representation
The reqwest load balancer middleware
- Host: GitHub
- URL: https://github.com/w-sodalite/reqwest-lb
- Owner: w-sodalite
- License: apache-2.0
- Created: 2024-10-19T08:25:59.000Z (3 months ago)
- Default Branch: main
- Last Pushed: 2024-12-04T10:57:22.000Z (about 1 month ago)
- Last Synced: 2024-12-04T11:35:27.335Z (about 1 month ago)
- Topics: load-balancer, reqwest, reqwest-middleware
- Language: Rust
- Homepage:
- Size: 38.1 KB
- Stars: 2
- Watchers: 1
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# reqwest-lb
[![Crates.io][crates-badge]][crates-url]
[![Apache licensed][apache-badge]][apache-url]
[![Build Status][actions-badge]][actions-url][crates-badge]: https://img.shields.io/crates/v/reqwest-lb.svg
[crates-url]: https://crates.io/crates/reqwest-lb
[apache-badge]: https://img.shields.io/badge/license-Aapche-blue.svg
[apache-url]: LICENSE
[actions-badge]: https://github.com/w-sodalite/reqwest-lb/workflows/CI/badge.svg
[actions-url]: https://github.com/w-sodalite/reqwest-lb/actions?query=workflow%3ACIA crate for [reqwest](https://crates.io/crates/reqwest) support load balancer, use
the [reqwest-middleware](https://crates.io/crates/reqwest-middleware).## Overview
This crate provide a middleware `LoadBalancerMiddleware`, it implement `reqwest-middleware::Middleware`, then use the `lb://` instead `http://` or `https://`。
- ### dependencies
```toml
[dependencies]
reqwest = "0.12"
reqwest-middleware = "0.3"
reqwest-lb = "0.1"
```- ### example
```rust
use reqwest::{Client, Url};
use reqwest_lb::LoadBalancerMiddleware;
use reqwest_lb::SimpleLoadBalancer;
use reqwest_lb::{LoadBalancerFactory, LoadBalancerPolicy};
use reqwest_middleware::{ClientBuilder, ClientWithMiddleware};
pub fn create(
ports: [u16; N],
policy: LoadBalancerPolicy,
) -> ClientWithMiddleware {
// create load balancer factory
let mut factory = LoadBalancerFactory::default();
// create url list
let mut urls = Vec::with_capacity(N);
for port in ports {
urls.push(Url::parse(format!("http://127.0.0.1:{}", port).as_str()).unwrap());
}
// create load balancer
let load_balancer = SimpleLoadBalancer::new(urls, policy);
// register load balancer for the host: example-server
factory.add("example-server", load_balancer);
// create reqwest client
let middleware = LoadBalancerMiddleware::new(factory);
ClientBuilder::new(Client::builder().no_proxy().build().unwrap())
.with(middleware)
.build()
}
pub async fn run() {
// use round robin policy
let client = create([3001, 3002], LoadBalancerPolicy::RoundRobin);
// http://127.0.0.1:3001
let response = client.get("lb://example-server/").send().await.unwrap();
// http://127.0.0.1:3002
let response = client.get("lb://example-server/").send().await.unwrap();
}
```- ### load balancer policy
- RoundRobin (default)
- Random
- First
- Last
- Weight- ### discovery
use `discovery` dynamic control the supplier elements, can send the elements change event `insert`,`remove` or
`initialized` to the stream.```rust
use http::Extensions;
use reqwest::Url;
use reqwest_lb::discovery::Change;
use reqwest_lb::supplier::discovery::DiscoverySupplier;
use reqwest_lb::{LoadBalancer, LoadBalancerPolicy, SimpleLoadBalancer};
#[tokio::test]
async fn laod_balancer_discovery() {
let ports = (0..10).map(|offset| 3000 + offset).collect::>();
let mut events = ports
.iter()
.map(|port| {
Url::parse(&format!("http://127.0.0.1:{}", port)).map(|url| Change::Insert(*port, url))
})
.collect::>();
events.push(Ok(Change::Initialized));
let discovery = futures::stream::iter(events);
let supplier = DiscoverySupplier::new(discovery);
let load_balancer = SimpleLoadBalancer::new(supplier, LoadBalancerPolicy::RoundRobin);
let mut extensions = Extensions::new();
for port in ports {
let selected = load_balancer.choose(&mut extensions).await;
assert_eq!(
selected,
Ok(Some(
Url::parse(&format!("http://127.0.0.1:{}", port)).unwrap()
))
);
}
}
```- ### with retry
> the load balancer policy use `RoundRobin`
```rust
use reqwest::{Client, Url};
use reqwest_lb::{
LoadBalancerFactory, LoadBalancerMiddleware, LoadBalancerPolicy, SimpleLoadBalancer,
};
use reqwest_middleware::ClientBuilder;
use reqwest_retry::policies::ExponentialBackoff;
use reqwest_retry::RetryTransientMiddleware;
#[tokio::test]
async fn with_retry() {
// load balancer middleware
let mut factory = LoadBalancerFactory::default();
let urls = vec![
// error
Url::parse("https://www.rust-lang-error.org").unwrap(),
// ok
Url::parse("https://www.rust-lang.org").unwrap(),
];
let load_balancer = SimpleLoadBalancer::new(urls, LoadBalancerPolicy::RoundRobin);
factory.add("rust-server", load_balancer);
let load_balancer_middleware = LoadBalancerMiddleware::new(factory);
// retry middleware
let retry_middleware = RetryTransientMiddleware::new_with_policy(
ExponentialBackoff::builder().build_with_max_retries(1),
);
// reqwest client
// https://www.rust-lang-error.org => retry error (choose next) => https://www.rust-lang.org
let client = ClientBuilder::new(Client::default())
.with(retry_middleware)
.with(load_balancer_middleware)
.build();
let response = client.get("lb://rust-server").send().await;
assert!(response.is_ok());
}
```## License
This project is licensed under the [Apache 2.0](./LICENSE)