fix(ap): add url field to Note, handle Delete(actor) and Tombstone objects
Some checks failed
lint / lint (push) Has been cancelled
test / unit (push) Has been cancelled
test / integration (push) Has been cancelled
lint / lint (pull_request) Failing after 9m12s
test / unit (pull_request) Successful in 15m52s
test / integration (pull_request) Failing after 17m10s
Some checks failed
lint / lint (push) Has been cancelled
test / unit (push) Has been cancelled
test / integration (push) Has been cancelled
lint / lint (pull_request) Failing after 9m12s
test / unit (pull_request) Successful in 15m52s
test / integration (pull_request) Failing after 17m10s
This commit is contained in:
@@ -337,7 +337,7 @@ pub struct DeleteActivity {
|
|||||||
#[serde(rename = "type", default)]
|
#[serde(rename = "type", default)]
|
||||||
pub(crate) kind: DeleteType,
|
pub(crate) kind: DeleteType,
|
||||||
pub(crate) actor: ObjectId<DbActor>,
|
pub(crate) actor: ObjectId<DbActor>,
|
||||||
pub(crate) object: Url,
|
pub(crate) object: serde_json::Value,
|
||||||
#[serde(skip_serializing_if = "Vec::is_empty", default)]
|
#[serde(skip_serializing_if = "Vec::is_empty", default)]
|
||||||
pub(crate) to: Vec<String>,
|
pub(crate) to: Vec<String>,
|
||||||
#[serde(skip_serializing_if = "Vec::is_empty", default)]
|
#[serde(skip_serializing_if = "Vec::is_empty", default)]
|
||||||
@@ -368,11 +368,38 @@ impl Activity for DeleteActivity {
|
|||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
let actor_url = self.actor.inner().clone();
|
let actor_url = self.actor.inner().clone();
|
||||||
|
|
||||||
|
// Extract object URL — handles plain string and Tombstone {"id":"...","type":"Tombstone"}
|
||||||
|
let object_url_str = match &self.object {
|
||||||
|
serde_json::Value::String(s) => s.clone(),
|
||||||
|
serde_json::Value::Object(o) => o
|
||||||
|
.get("id")
|
||||||
|
.and_then(|v| v.as_str())
|
||||||
|
.map(|s| s.to_string())
|
||||||
|
.unwrap_or_default(),
|
||||||
|
_ => String::new(),
|
||||||
|
};
|
||||||
|
let Ok(object_url) = Url::parse(&object_url_str) else {
|
||||||
|
tracing::warn!(actor = %actor_url, "Delete activity has unparseable object, ignoring");
|
||||||
|
return Ok(());
|
||||||
|
};
|
||||||
|
|
||||||
|
// Actor self-deletion: Mastodon sends Delete(actor_url) when an account is deleted.
|
||||||
|
if object_url == *self.actor.inner() {
|
||||||
|
data.object_handler
|
||||||
|
.on_actor_removed(&actor_url)
|
||||||
|
.await
|
||||||
|
.map_err(|e| Error::from(anyhow::anyhow!(e)))?;
|
||||||
|
tracing::info!(actor = %actor_url, "received Delete(actor) — remote account deleted");
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Normal note deletion.
|
||||||
data.object_handler
|
data.object_handler
|
||||||
.on_delete(&self.object, &actor_url)
|
.on_delete(&object_url, &actor_url)
|
||||||
.await
|
.await
|
||||||
.map_err(|e| Error::from(anyhow::anyhow!(e)))?;
|
.map_err(|e| Error::from(anyhow::anyhow!(e)))?;
|
||||||
tracing::info!(object = %self.object, "received delete activity");
|
tracing::info!(object = %object_url, "received Delete(note)");
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -39,6 +39,7 @@ fn thought_note_json(
|
|||||||
let mut note = serde_json::json!({
|
let mut note = serde_json::json!({
|
||||||
"type": "Note",
|
"type": "Note",
|
||||||
"id": ap_id.to_string(),
|
"id": ap_id.to_string(),
|
||||||
|
"url": ap_id.to_string(),
|
||||||
"attributedTo": local_actor.ap_id.to_string(),
|
"attributedTo": local_actor.ap_id.to_string(),
|
||||||
"content": thought.content.as_str(),
|
"content": thought.content.as_str(),
|
||||||
"published": thought.created_at.to_rfc3339(),
|
"published": thought.created_at.to_rfc3339(),
|
||||||
@@ -653,7 +654,7 @@ impl ActivityPubService {
|
|||||||
id: delete_id,
|
id: delete_id,
|
||||||
kind: Default::default(),
|
kind: Default::default(),
|
||||||
actor: ObjectId::from(local_actor.ap_id.clone()),
|
actor: ObjectId::from(local_actor.ap_id.clone()),
|
||||||
object: ap_id,
|
object: serde_json::json!(ap_id.to_string()),
|
||||||
to: vec![crate::urls::AS_PUBLIC.to_string()],
|
to: vec![crate::urls::AS_PUBLIC.to_string()],
|
||||||
cc: vec![local_actor.followers_url.to_string()],
|
cc: vec![local_actor.followers_url.to_string()],
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ pub struct ThoughtNote {
|
|||||||
#[serde(rename = "type")]
|
#[serde(rename = "type")]
|
||||||
pub kind: NoteType,
|
pub kind: NoteType,
|
||||||
pub id: Url,
|
pub id: Url,
|
||||||
|
pub url: Url, // Mastodon uses this as the clickable link
|
||||||
pub attributed_to: Url,
|
pub attributed_to: Url,
|
||||||
pub content: String,
|
pub content: String,
|
||||||
pub published: DateTime<Utc>,
|
pub published: DateTime<Utc>,
|
||||||
@@ -39,6 +40,7 @@ impl ThoughtNote {
|
|||||||
) -> Self {
|
) -> Self {
|
||||||
Self {
|
Self {
|
||||||
kind: Default::default(),
|
kind: Default::default(),
|
||||||
|
url: id.clone(),
|
||||||
id,
|
id,
|
||||||
attributed_to: actor_url,
|
attributed_to: actor_url,
|
||||||
content,
|
content,
|
||||||
@@ -71,5 +73,6 @@ mod tests {
|
|||||||
let json = serde_json::to_string(¬e).unwrap();
|
let json = serde_json::to_string(¬e).unwrap();
|
||||||
assert!(json.contains(AS_PUBLIC));
|
assert!(json.contains(AS_PUBLIC));
|
||||||
assert!(json.contains("Hello world"));
|
assert!(json.contains("Hello world"));
|
||||||
|
assert!(json.contains("\"url\""));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user