summaryrefslogtreecommitdiffstats
path: root/server/src/request.rs
blob: 7d09b60df99ac820f0aadfa4f7e90a5bf470b2f1 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
use crate::LemmyError;
use std::future::Future;

#[derive(Clone, Debug, Fail)]
#[fail(display = "Error sending request, {}", _0)]
struct SendError(pub String);

#[derive(Clone, Debug, Fail)]
#[fail(display = "Error receiving response, {}", _0)]
pub struct RecvError(pub String);

pub async fn retry<F, Fut, T>(f: F) -> Result<T, LemmyError>
where
  F: Fn() -> Fut,
  Fut: Future<Output = Result<T, actix_web::client::SendRequestError>>,
{
  retry_custom(|| async { Ok((f)().await) }).await
}

pub async fn retry_custom<F, Fut, T>(f: F) -> Result<T, LemmyError>
where
  F: Fn() -> Fut,
  Fut: Future<Output = Result<Result<T, actix_web::client::SendRequestError>, LemmyError>>,
{
  let mut response = Err(format_err!("connect timeout").into());

  for _ in 0u8..3 {
    match (f)().await? {
      Ok(t) => return Ok(t),
      Err(e) => {
        if is_connect_timeout(&e) {
          response = Err(SendError(e.to_string()).into());
          continue;
        }
        return Err(SendError(e.to_string()).into());
      }
    }
  }

  response
}

fn is_connect_timeout(e: &actix_web::client::SendRequestError) -> bool {
  if let actix_web::client::SendRequestError::Connect(e) = e {
    if let actix_web::client::ConnectError::Timeout = e {
      return true;
    }
  }

  false
}