feat: add fetch_outbox on folreq and outbox?fetch

This commit is contained in:
əlemi 2025-01-09 20:13:25 +01:00
parent 9b6f438a98
commit 815fdcd883
Signed by: alemi
GPG key ID: A4895B84D311642C
3 changed files with 28 additions and 2 deletions

View file

@ -101,6 +101,7 @@ pub trait Fetcher {
async fn resolve_object(&self, object: serde_json::Value, tx: &impl ConnectionTrait) -> Result<crate::model::object::Model, RequestError>; async fn resolve_object(&self, object: serde_json::Value, tx: &impl ConnectionTrait) -> Result<crate::model::object::Model, RequestError>;
async fn fetch_thread(&self, id: &str, tx: &impl ConnectionTrait) -> Result<(), RequestError>; async fn fetch_thread(&self, id: &str, tx: &impl ConnectionTrait) -> Result<(), RequestError>;
async fn fetch_outbox(&self, id: &str, tx: &impl ConnectionTrait) -> Result<(), RequestError>;
fn client(domain: &str) -> reqwest::Client { fn client(domain: &str) -> reqwest::Client {
reqwest::Client::builder() reqwest::Client::builder()
@ -447,6 +448,8 @@ impl Fetcher for crate::Context {
page = page.first().into_inner()?; page = page.first().into_inner()?;
} }
// TODO parallelize these
for obj in page.items().flat() { for obj in page.items().flat() {
if let Err(e) = self.fetch_object(&obj.id()?, tx).await { if let Err(e) = self.fetch_object(&obj.id()?, tx).await {
tracing::warn!("error fetching reply: {e}"); tracing::warn!("error fetching reply: {e}");
@ -466,6 +469,21 @@ impl Fetcher for crate::Context {
Ok(()) Ok(())
} }
async fn fetch_outbox(&self, id: &str, tx: &impl ConnectionTrait) -> Result<(), RequestError> {
let actor = self.pull(id).await?.actor()?;
let outbox = actor
.outbox().resolve(self).await?
.first().resolve(self).await?;
// TODO parallelize these
for item in outbox.ordered_items().all_ids() {
self.fetch_object(&item, tx).await?;
}
Ok(())
}
async fn fetch_object(&self, id: &str, tx: &impl ConnectionTrait) -> Result<crate::model::object::Model, RequestError> { async fn fetch_object(&self, id: &str, tx: &impl ConnectionTrait) -> Result<crate::model::object::Model, RequestError> {
fetch_object_r(self, id, 0, tx).await fetch_object_r(self, id, 0, tx).await
} }

View file

@ -203,6 +203,7 @@ pub async fn follow(ctx: &crate::Context, activity: impl apb::Activity, tx: &Dat
ctx.address(Some(&activity_model), None, tx).await?; ctx.address(Some(&activity_model), None, tx).await?;
if ctx.is_local(&target_actor.id) { if ctx.is_local(&target_actor.id) {
ctx.fetch_outbox(&target_actor.id, tx).await?;
crate::Query::notify(activity_model.internal, target_actor.internal) crate::Query::notify(activity_model.internal, target_actor.internal)
.exec(tx) .exec(tx)
.await?; .await?;

View file

@ -1,14 +1,21 @@
use axum::{extract::{Path, Query, State}, http::StatusCode, Json}; use axum::{extract::{Path, Query, State}, http::StatusCode, Json};
use sea_orm::{ActiveValue::{NotSet, Set}, ColumnTrait, Condition, EntityTrait, QueryFilter, QueryOrder, QuerySelect}; use sea_orm::{ActiveValue::{NotSet, Set}, ColumnTrait, Condition, EntityTrait, QueryFilter, QueryOrder, QuerySelect};
use upub::{model, selector::{RichActivity, RichFillable}, Context}; use upub::{model, selector::{RichActivity, RichFillable}, traits::Fetcher, Context};
use crate::{activitypub::{CreationResult, Pagination}, builders::JsonLD, AuthIdentity, Identity}; use crate::{activitypub::{CreationResult, Pagination, TryFetch}, builders::JsonLD, AuthIdentity, Identity};
pub async fn get( pub async fn get(
State(ctx): State<Context>, State(ctx): State<Context>,
AuthIdentity(auth): AuthIdentity,
Path(id): Path<String>, Path(id): Path<String>,
Query(query): Query<TryFetch>,
) -> crate::ApiResult<JsonLD<serde_json::Value>> { ) -> crate::ApiResult<JsonLD<serde_json::Value>> {
let uid = ctx.uid(&id);
if auth.is_local() && query.fetch && !ctx.is_local(&uid) {
ctx.fetch_outbox(&uid, ctx.db()).await?;
}
crate::builders::collection(upub::url!(ctx, "/actors/{id}/outbox"), None) crate::builders::collection(upub::url!(ctx, "/actors/{id}/outbox"), None)
} }