2020-03-30 17:10:04 +00:00
|
|
|
use crate::{
|
2020-05-21 21:24:56 +00:00
|
|
|
apub::AcceptedActivities,
|
2020-03-30 17:10:04 +00:00
|
|
|
config::{Config, UrlKind},
|
2021-02-10 04:05:06 +00:00
|
|
|
db::Actor,
|
2020-03-30 17:10:04 +00:00
|
|
|
error::MyError,
|
|
|
|
jobs::{apub::prepare_activity, Deliver, JobState},
|
|
|
|
};
|
2020-09-07 21:51:02 +00:00
|
|
|
use activitystreams::{
|
2020-05-21 21:24:56 +00:00
|
|
|
activity::{Accept as AsAccept, Follow as AsFollow},
|
|
|
|
prelude::*,
|
2020-06-20 04:11:02 +00:00
|
|
|
url::Url,
|
2020-05-21 21:24:56 +00:00
|
|
|
};
|
2020-04-21 00:56:50 +00:00
|
|
|
use background_jobs::ActixJob;
|
2020-03-30 17:10:04 +00:00
|
|
|
use std::{future::Future, pin::Pin};
|
|
|
|
|
|
|
|
#[derive(Clone, Debug, serde::Deserialize, serde::Serialize)]
|
|
|
|
pub struct Follow {
|
|
|
|
is_listener: bool,
|
2020-05-21 21:24:56 +00:00
|
|
|
input: AcceptedActivities,
|
2020-03-30 17:10:04 +00:00
|
|
|
actor: Actor,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Follow {
|
2020-05-21 21:24:56 +00:00
|
|
|
pub fn new(is_listener: bool, input: AcceptedActivities, actor: Actor) -> Self {
|
2020-03-30 17:10:04 +00:00
|
|
|
Follow {
|
|
|
|
is_listener,
|
|
|
|
input,
|
|
|
|
actor,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn perform(self, state: JobState) -> Result<(), anyhow::Error> {
|
|
|
|
if !self.is_listener {
|
2020-06-27 22:29:23 +00:00
|
|
|
state.db.add_listener(self.actor.inbox.clone()).await?;
|
2020-03-30 17:10:04 +00:00
|
|
|
}
|
2020-06-03 17:37:36 +00:00
|
|
|
let my_id = state.config.generate_url(UrlKind::Actor);
|
2020-03-30 17:10:04 +00:00
|
|
|
|
|
|
|
// if following relay directly, not just following 'public', followback
|
2021-02-10 04:05:06 +00:00
|
|
|
if self.input.object_is(&my_id)
|
|
|
|
&& !state.state.db.is_connected(self.actor.id.clone()).await?
|
|
|
|
{
|
2020-03-30 17:10:04 +00:00
|
|
|
let follow = generate_follow(&state.config, &self.actor.id, &my_id)?;
|
|
|
|
state
|
|
|
|
.job_server
|
|
|
|
.queue(Deliver::new(self.actor.inbox.clone(), follow)?)?;
|
|
|
|
}
|
|
|
|
|
2021-02-10 04:05:06 +00:00
|
|
|
state.actors.follower(self.actor.clone()).await?;
|
2020-03-30 17:10:04 +00:00
|
|
|
|
2020-05-21 21:24:56 +00:00
|
|
|
let accept = generate_accept_follow(
|
|
|
|
&state.config,
|
|
|
|
&self.actor.id,
|
2020-06-20 04:11:02 +00:00
|
|
|
self.input.id_unchecked().ok_or(MyError::MissingId)?,
|
2020-05-21 21:24:56 +00:00
|
|
|
&my_id,
|
|
|
|
)?;
|
2020-03-30 17:10:04 +00:00
|
|
|
|
|
|
|
state
|
|
|
|
.job_server
|
|
|
|
.queue(Deliver::new(self.actor.inbox, accept)?)?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate a type that says "I want to follow you"
|
2020-06-20 04:11:02 +00:00
|
|
|
fn generate_follow(config: &Config, actor_id: &Url, my_id: &Url) -> Result<AsFollow, MyError> {
|
2020-05-21 21:24:56 +00:00
|
|
|
let follow = AsFollow::new(my_id.clone(), actor_id.clone());
|
2020-03-30 17:10:04 +00:00
|
|
|
|
|
|
|
prepare_activity(
|
|
|
|
follow,
|
|
|
|
config.generate_url(UrlKind::Activity),
|
|
|
|
actor_id.clone(),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate a type that says "I accept your follow request"
|
|
|
|
fn generate_accept_follow(
|
|
|
|
config: &Config,
|
2020-06-20 04:11:02 +00:00
|
|
|
actor_id: &Url,
|
|
|
|
input_id: &Url,
|
|
|
|
my_id: &Url,
|
2020-05-21 21:24:56 +00:00
|
|
|
) -> Result<AsAccept, MyError> {
|
|
|
|
let mut follow = AsFollow::new(actor_id.clone(), my_id.clone());
|
|
|
|
|
|
|
|
follow.set_id(input_id.clone());
|
|
|
|
|
|
|
|
let accept = AsAccept::new(my_id.clone(), follow.into_any_base()?);
|
2020-03-30 17:10:04 +00:00
|
|
|
|
|
|
|
prepare_activity(
|
|
|
|
accept,
|
|
|
|
config.generate_url(UrlKind::Activity),
|
|
|
|
actor_id.clone(),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ActixJob for Follow {
|
|
|
|
type State = JobState;
|
|
|
|
type Future = Pin<Box<dyn Future<Output = Result<(), anyhow::Error>>>>;
|
|
|
|
|
2020-04-21 01:03:46 +00:00
|
|
|
const NAME: &'static str = "relay::jobs::apub::Follow";
|
2020-04-21 00:56:50 +00:00
|
|
|
|
2020-03-30 17:10:04 +00:00
|
|
|
fn run(self, state: Self::State) -> Self::Future {
|
|
|
|
Box::pin(self.perform(state))
|
|
|
|
}
|
|
|
|
}
|