mirror of
https://git.asonix.dog/asonix/relay.git
synced 2025-01-08 18:45:24 +00:00
Enable different breaker failure cases for different endpoints
Additionally, don't count 4xx towards succeeding a breaker
This commit is contained in:
parent
5a6fbbcb77
commit
804d22ee81
8 changed files with 114 additions and 27 deletions
|
@ -2,7 +2,7 @@ use crate::{
|
|||
apub::AcceptedActors,
|
||||
db::{Actor, Db},
|
||||
error::{Error, ErrorKind},
|
||||
requests::Requests,
|
||||
requests::{BreakerStrategy, Requests},
|
||||
};
|
||||
use activitystreams::{iri_string::types::IriString, prelude::*};
|
||||
use std::time::{Duration, SystemTime};
|
||||
|
@ -71,7 +71,9 @@ impl ActorCache {
|
|||
id: &IriString,
|
||||
requests: &Requests,
|
||||
) -> Result<Actor, Error> {
|
||||
let accepted_actor = requests.fetch::<AcceptedActors>(id).await?;
|
||||
let accepted_actor = requests
|
||||
.fetch::<AcceptedActors>(id, BreakerStrategy::Require2XX)
|
||||
.await?;
|
||||
|
||||
let input_authority = id.authority_components().ok_or(ErrorKind::MissingDomain)?;
|
||||
let accepted_actor_id = accepted_actor
|
||||
|
|
|
@ -2,6 +2,7 @@ use crate::{
|
|||
apub::AcceptedActors,
|
||||
error::{Error, ErrorKind},
|
||||
jobs::JobState,
|
||||
requests::BreakerStrategy,
|
||||
};
|
||||
use activitystreams::{iri_string::types::IriString, object::Image, prelude::*};
|
||||
use background_jobs::ActixJob;
|
||||
|
@ -44,7 +45,7 @@ impl QueryContact {
|
|||
let contact = match state
|
||||
.state
|
||||
.requests
|
||||
.fetch::<AcceptedActors>(&self.contact_id)
|
||||
.fetch::<AcceptedActors>(&self.contact_id, BreakerStrategy::Allow404AndBelow)
|
||||
.await
|
||||
{
|
||||
Ok(contact) => contact,
|
||||
|
|
|
@ -1,6 +1,7 @@
|
|||
use crate::{
|
||||
error::Error,
|
||||
jobs::{debug_object, JobState},
|
||||
requests::BreakerStrategy,
|
||||
};
|
||||
use activitystreams::iri_string::types::IriString;
|
||||
use background_jobs::{ActixJob, Backoff};
|
||||
|
@ -35,7 +36,12 @@ impl Deliver {
|
|||
|
||||
#[tracing::instrument(name = "Deliver", skip(state))]
|
||||
async fn permform(self, state: JobState) -> Result<(), Error> {
|
||||
if let Err(e) = state.state.requests.deliver(&self.to, &self.data).await {
|
||||
if let Err(e) = state
|
||||
.state
|
||||
.requests
|
||||
.deliver(&self.to, &self.data, BreakerStrategy::Allow401AndBelow)
|
||||
.await
|
||||
{
|
||||
if e.is_breaker() {
|
||||
tracing::debug!("Not trying due to failed breaker");
|
||||
return Ok(());
|
||||
|
|
|
@ -2,6 +2,7 @@ use crate::{
|
|||
config::UrlKind,
|
||||
error::{Error, ErrorKind},
|
||||
jobs::{Boolish, JobState},
|
||||
requests::BreakerStrategy,
|
||||
};
|
||||
use activitystreams::{iri, iri_string::types::IriString};
|
||||
use background_jobs::ActixJob;
|
||||
|
@ -42,7 +43,10 @@ impl QueryInstance {
|
|||
state
|
||||
.state
|
||||
.requests
|
||||
.fetch_json::<Instance>(&mastodon_instance_uri)
|
||||
.fetch_json::<Instance>(
|
||||
&mastodon_instance_uri,
|
||||
BreakerStrategy::Allow404AndBelow,
|
||||
)
|
||||
.await
|
||||
}
|
||||
InstanceApiType::Misskey => {
|
||||
|
@ -50,7 +54,10 @@ impl QueryInstance {
|
|||
state
|
||||
.state
|
||||
.requests
|
||||
.fetch_json_msky::<MisskeyMeta>(&msky_meta_uri)
|
||||
.fetch_json_msky::<MisskeyMeta>(
|
||||
&msky_meta_uri,
|
||||
BreakerStrategy::Allow404AndBelow,
|
||||
)
|
||||
.await
|
||||
.map(|res| res.into())
|
||||
}
|
||||
|
|
|
@ -1,6 +1,7 @@
|
|||
use crate::{
|
||||
error::{Error, ErrorKind},
|
||||
jobs::{Boolish, JobState, QueryContact},
|
||||
requests::BreakerStrategy,
|
||||
};
|
||||
use activitystreams::{iri, iri_string::types::IriString, primitives::OneOrMany};
|
||||
use background_jobs::ActixJob;
|
||||
|
@ -45,7 +46,7 @@ impl QueryNodeinfo {
|
|||
let well_known = match state
|
||||
.state
|
||||
.requests
|
||||
.fetch_json::<WellKnown>(&well_known_uri)
|
||||
.fetch_json::<WellKnown>(&well_known_uri, BreakerStrategy::Allow404AndBelow)
|
||||
.await
|
||||
{
|
||||
Ok(well_known) => well_known,
|
||||
|
@ -62,7 +63,12 @@ impl QueryNodeinfo {
|
|||
return Ok(());
|
||||
};
|
||||
|
||||
let nodeinfo = match state.state.requests.fetch_json::<Nodeinfo>(&href).await {
|
||||
let nodeinfo = match state
|
||||
.state
|
||||
.requests
|
||||
.fetch_json::<Nodeinfo>(&href, BreakerStrategy::Require2XX)
|
||||
.await
|
||||
{
|
||||
Ok(nodeinfo) => nodeinfo,
|
||||
Err(e) if e.is_breaker() => {
|
||||
tracing::debug!("Not retrying due to failed breaker");
|
||||
|
|
|
@ -2,7 +2,7 @@ use crate::{
|
|||
apub::AcceptedActors,
|
||||
data::{ActorCache, State},
|
||||
error::{Error, ErrorKind},
|
||||
requests::Requests,
|
||||
requests::{BreakerStrategy, Requests},
|
||||
spawner::Spawner,
|
||||
};
|
||||
use activitystreams::{base::BaseExt, iri, iri_string::types::IriString};
|
||||
|
@ -70,7 +70,11 @@ impl MyVerify {
|
|||
|
||||
actor_id
|
||||
} else {
|
||||
match self.0.fetch::<PublicKeyResponse>(&public_key_id).await {
|
||||
match self
|
||||
.0
|
||||
.fetch::<PublicKeyResponse>(&public_key_id, BreakerStrategy::Require2XX)
|
||||
.await
|
||||
{
|
||||
Ok(res) => res.actor_id().ok_or(ErrorKind::MissingId),
|
||||
Err(e) => {
|
||||
if e.is_gone() {
|
||||
|
|
|
@ -24,6 +24,16 @@ const ONE_MINUTE: u64 = 60 * ONE_SECOND;
|
|||
const ONE_HOUR: u64 = 60 * ONE_MINUTE;
|
||||
const ONE_DAY: u64 = 24 * ONE_HOUR;
|
||||
|
||||
#[derive(Debug)]
|
||||
pub(crate) enum BreakerStrategy {
|
||||
// Requires a successful response
|
||||
Require2XX,
|
||||
// Allows HTTP 2xx-401
|
||||
Allow401AndBelow,
|
||||
// Allows HTTP 2xx-404
|
||||
Allow404AndBelow,
|
||||
}
|
||||
|
||||
#[derive(Clone)]
|
||||
pub(crate) struct Breakers {
|
||||
inner: Arc<DashMap<String, Breaker>>,
|
||||
|
@ -193,6 +203,7 @@ impl Requests {
|
|||
async fn check_response(
|
||||
&self,
|
||||
parsed_url: &IriString,
|
||||
strategy: BreakerStrategy,
|
||||
res: Result<reqwest::Response, reqwest_middleware::Error>,
|
||||
) -> Result<reqwest::Response, Error> {
|
||||
if res.is_err() {
|
||||
|
@ -203,7 +214,13 @@ impl Requests {
|
|||
|
||||
let status = res.status();
|
||||
|
||||
if status.is_server_error() {
|
||||
let success = match strategy {
|
||||
BreakerStrategy::Require2XX => status.is_success(),
|
||||
BreakerStrategy::Allow401AndBelow => (200..=401).contains(&status.as_u16()),
|
||||
BreakerStrategy::Allow404AndBelow => (200..=404).contains(&status.as_u16()),
|
||||
};
|
||||
|
||||
if !success {
|
||||
self.breakers.fail(&parsed_url);
|
||||
|
||||
if let Ok(s) = res.text().await {
|
||||
|
@ -215,22 +232,33 @@ impl Requests {
|
|||
return Err(ErrorKind::Status(parsed_url.to_string(), status).into());
|
||||
}
|
||||
|
||||
self.last_online.mark_seen(&parsed_url);
|
||||
self.breakers.succeed(&parsed_url);
|
||||
// only actually succeed a breaker on 2xx response
|
||||
if status.is_success() {
|
||||
self.last_online.mark_seen(&parsed_url);
|
||||
self.breakers.succeed(&parsed_url);
|
||||
}
|
||||
|
||||
Ok(res)
|
||||
}
|
||||
|
||||
#[tracing::instrument(name = "Fetch Json", skip(self), fields(signing_string))]
|
||||
pub(crate) async fn fetch_json<T>(&self, url: &IriString) -> Result<T, Error>
|
||||
pub(crate) async fn fetch_json<T>(
|
||||
&self,
|
||||
url: &IriString,
|
||||
strategy: BreakerStrategy,
|
||||
) -> Result<T, Error>
|
||||
where
|
||||
T: serde::de::DeserializeOwned,
|
||||
{
|
||||
self.do_fetch(url, "application/json").await
|
||||
self.do_fetch(url, "application/json", strategy).await
|
||||
}
|
||||
|
||||
#[tracing::instrument(name = "Fetch Json", skip(self), fields(signing_string))]
|
||||
pub(crate) async fn fetch_json_msky<T>(&self, url: &IriString) -> Result<T, Error>
|
||||
pub(crate) async fn fetch_json_msky<T>(
|
||||
&self,
|
||||
url: &IriString,
|
||||
strategy: BreakerStrategy,
|
||||
) -> Result<T, Error>
|
||||
where
|
||||
T: serde::de::DeserializeOwned,
|
||||
{
|
||||
|
@ -240,6 +268,7 @@ impl Requests {
|
|||
&serde_json::json!({}),
|
||||
"application/json",
|
||||
"application/json",
|
||||
strategy,
|
||||
)
|
||||
.await?
|
||||
.bytes()
|
||||
|
@ -249,31 +278,50 @@ impl Requests {
|
|||
}
|
||||
|
||||
#[tracing::instrument(name = "Fetch Activity+Json", skip(self), fields(signing_string))]
|
||||
pub(crate) async fn fetch<T>(&self, url: &IriString) -> Result<T, Error>
|
||||
pub(crate) async fn fetch<T>(
|
||||
&self,
|
||||
url: &IriString,
|
||||
strategy: BreakerStrategy,
|
||||
) -> Result<T, Error>
|
||||
where
|
||||
T: serde::de::DeserializeOwned,
|
||||
{
|
||||
self.do_fetch(url, "application/activity+json").await
|
||||
self.do_fetch(url, "application/activity+json", strategy)
|
||||
.await
|
||||
}
|
||||
|
||||
async fn do_fetch<T>(&self, url: &IriString, accept: &str) -> Result<T, Error>
|
||||
async fn do_fetch<T>(
|
||||
&self,
|
||||
url: &IriString,
|
||||
accept: &str,
|
||||
strategy: BreakerStrategy,
|
||||
) -> Result<T, Error>
|
||||
where
|
||||
T: serde::de::DeserializeOwned,
|
||||
{
|
||||
let body = self.do_fetch_response(url, accept).await?.bytes().await?;
|
||||
let body = self
|
||||
.do_fetch_response(url, accept, strategy)
|
||||
.await?
|
||||
.bytes()
|
||||
.await?;
|
||||
|
||||
Ok(serde_json::from_slice(&body)?)
|
||||
}
|
||||
|
||||
#[tracing::instrument(name = "Fetch response", skip(self), fields(signing_string))]
|
||||
pub(crate) async fn fetch_response(&self, url: &IriString) -> Result<reqwest::Response, Error> {
|
||||
self.do_fetch_response(url, "*/*").await
|
||||
pub(crate) async fn fetch_response(
|
||||
&self,
|
||||
url: &IriString,
|
||||
strategy: BreakerStrategy,
|
||||
) -> Result<reqwest::Response, Error> {
|
||||
self.do_fetch_response(url, "*/*", strategy).await
|
||||
}
|
||||
|
||||
pub(crate) async fn do_fetch_response(
|
||||
&self,
|
||||
url: &IriString,
|
||||
accept: &str,
|
||||
strategy: BreakerStrategy,
|
||||
) -> Result<reqwest::Response, Error> {
|
||||
if !self.breakers.should_try(url) {
|
||||
return Err(ErrorKind::Breaker.into());
|
||||
|
@ -295,7 +343,7 @@ impl Requests {
|
|||
|
||||
let res = self.client.execute(request).await;
|
||||
|
||||
let res = self.check_response(url, res).await?;
|
||||
let res = self.check_response(url, strategy, res).await?;
|
||||
|
||||
Ok(res)
|
||||
}
|
||||
|
@ -305,7 +353,12 @@ impl Requests {
|
|||
skip_all,
|
||||
fields(inbox = inbox.to_string().as_str(), signing_string)
|
||||
)]
|
||||
pub(crate) async fn deliver<T>(&self, inbox: &IriString, item: &T) -> Result<(), Error>
|
||||
pub(crate) async fn deliver<T>(
|
||||
&self,
|
||||
inbox: &IriString,
|
||||
item: &T,
|
||||
strategy: BreakerStrategy,
|
||||
) -> Result<(), Error>
|
||||
where
|
||||
T: serde::ser::Serialize + std::fmt::Debug,
|
||||
{
|
||||
|
@ -314,6 +367,7 @@ impl Requests {
|
|||
item,
|
||||
"application/activity+json",
|
||||
"application/activity+json",
|
||||
strategy,
|
||||
)
|
||||
.await?;
|
||||
Ok(())
|
||||
|
@ -325,6 +379,7 @@ impl Requests {
|
|||
item: &T,
|
||||
content_type: &str,
|
||||
accept: &str,
|
||||
strategy: BreakerStrategy,
|
||||
) -> Result<reqwest::Response, Error>
|
||||
where
|
||||
T: serde::ser::Serialize + std::fmt::Debug,
|
||||
|
@ -357,7 +412,7 @@ impl Requests {
|
|||
|
||||
let res = self.client.execute(request).await;
|
||||
|
||||
let res = self.check_response(inbox, res).await?;
|
||||
let res = self.check_response(inbox, strategy, res).await?;
|
||||
|
||||
Ok(res)
|
||||
}
|
||||
|
|
|
@ -1,4 +1,8 @@
|
|||
use crate::{data::MediaCache, error::Error, requests::Requests};
|
||||
use crate::{
|
||||
data::MediaCache,
|
||||
error::Error,
|
||||
requests::{BreakerStrategy, Requests},
|
||||
};
|
||||
use actix_web::{body::BodyStream, web, HttpResponse};
|
||||
use uuid::Uuid;
|
||||
|
||||
|
@ -11,7 +15,9 @@ pub(crate) async fn route(
|
|||
let uuid = uuid.into_inner();
|
||||
|
||||
if let Some(url) = media.get_url(uuid).await? {
|
||||
let res = requests.fetch_response(&url).await?;
|
||||
let res = requests
|
||||
.fetch_response(&url, BreakerStrategy::Allow404AndBelow)
|
||||
.await?;
|
||||
|
||||
let mut response = HttpResponse::build(res.status());
|
||||
|
||||
|
|
Loading…
Reference in a new issue