Undo(Announce): now removes announce record from ActorRepository and
calls ApObjectHandler::on_announce_removed (default no-op, override
to decrement boost counts). Announce counts no longer drift.
Undo(Block): now logged at info level instead of silently ignored.
No automatic relationship restoration (spec doesn't require it).
AddActivity: now uses object["id"] as the stable ap_id (same as
CreateActivity), falling back to activity id only if object has no
id field. Fixes keying watchlist/collection items by the wrong id.
Featured collection: GET /users/{id}/featured now served by the router.
ApContentReader::get_featured_objects() has a default empty-list impl
— override to expose pinned posts without any breaking changes.
74 lines
2.3 KiB
Rust
74 lines
2.3 KiB
Rust
use activitypub_federation::{config::Data, fetch::object_id::ObjectId, traits::Activity};
|
|
use serde::{Deserialize, Serialize};
|
|
use url::Url;
|
|
|
|
use crate::actors::DbActor;
|
|
use crate::data::FederationData;
|
|
use crate::error::Error;
|
|
|
|
use super::helpers::check_guards;
|
|
|
|
#[derive(Clone, Default, Debug, Serialize, Deserialize)]
|
|
#[serde(rename = "Add")]
|
|
pub struct AddType;
|
|
|
|
#[derive(Debug, Deserialize, Serialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub struct AddActivity {
|
|
pub(crate) id: Url,
|
|
#[serde(rename = "type", default)]
|
|
pub(crate) kind: AddType,
|
|
pub(crate) actor: ObjectId<DbActor>,
|
|
pub(crate) object: serde_json::Value,
|
|
#[serde(skip_serializing_if = "Vec::is_empty", default)]
|
|
pub(crate) to: Vec<String>,
|
|
#[serde(skip_serializing_if = "Vec::is_empty", default)]
|
|
pub(crate) cc: Vec<String>,
|
|
}
|
|
|
|
#[async_trait::async_trait]
|
|
impl Activity for AddActivity {
|
|
type DataType = FederationData;
|
|
type Error = Error;
|
|
|
|
fn id(&self) -> &Url {
|
|
&self.id
|
|
}
|
|
fn actor(&self) -> &Url {
|
|
self.actor.inner()
|
|
}
|
|
|
|
async fn verify(&self, _data: &Data<Self::DataType>) -> Result<(), Self::Error> {
|
|
if let Some(attributed_to) = self.object.get("attributedTo").and_then(|v| v.as_str())
|
|
&& let Ok(attributed_url) = Url::parse(attributed_to)
|
|
&& &attributed_url != self.actor.inner()
|
|
{
|
|
return Err(Error::bad_request(anyhow::anyhow!(
|
|
"Add actor does not match object attributedTo"
|
|
)));
|
|
}
|
|
Ok(())
|
|
}
|
|
|
|
async fn receive(self, data: &Data<Self::DataType>) -> Result<(), Self::Error> {
|
|
if check_guards(&self.id, self.actor.inner(), data).await? {
|
|
return Ok(());
|
|
}
|
|
// Use the object's own id as the stable AP identifier, falling back to
|
|
// the activity id only if the object has no id field.
|
|
let ap_id = self
|
|
.object
|
|
.get("id")
|
|
.and_then(|v| v.as_str())
|
|
.and_then(|s| Url::parse(s).ok())
|
|
.unwrap_or_else(|| self.id.clone());
|
|
let actor_url = self.actor.inner().clone();
|
|
data.object_handler
|
|
.on_create(&ap_id, &actor_url, self.object)
|
|
.await
|
|
.map_err(|e| Error::from(anyhow::anyhow!(e)))?;
|
|
tracing::info!(actor = %actor_url, "received Add activity");
|
|
Ok(())
|
|
}
|
|
}
|