feat(domain): ActivityPubRepository port with federation vocabulary
This commit is contained in:
@@ -13,6 +13,7 @@ uuid = { workspace = true }
|
|||||||
chrono = { workspace = true }
|
chrono = { workspace = true }
|
||||||
serde = { workspace = true }
|
serde = { workspace = true }
|
||||||
futures = { workspace = true }
|
futures = { workspace = true }
|
||||||
|
url = { workspace = true }
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
tokio = { workspace = true, features = ["full"] }
|
tokio = { workspace = true, features = ["full"] }
|
||||||
|
|||||||
@@ -154,3 +154,82 @@ pub trait SearchPort: Send + Sync {
|
|||||||
page: &PageParams,
|
page: &PageParams,
|
||||||
) -> Result<Paginated<User>, DomainError>;
|
) -> Result<Paginated<User>, DomainError>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// A local thought ready for AP serialization, with the author's username
|
||||||
|
/// pre-joined so the handler can build AP URLs without a second query.
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct OutboxEntry {
|
||||||
|
pub thought: crate::models::thought::Thought,
|
||||||
|
pub author_username: Username,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
pub trait ActivityPubRepository: Send + Sync {
|
||||||
|
// ── Outbox (local → remote) ──────────────────────────────────────
|
||||||
|
|
||||||
|
/// All public local thoughts for this actor. Used for outbox totals
|
||||||
|
/// and full-collection delivery.
|
||||||
|
async fn outbox_entries_for_actor(
|
||||||
|
&self,
|
||||||
|
user_id: &UserId,
|
||||||
|
) -> Result<Vec<OutboxEntry>, DomainError>;
|
||||||
|
|
||||||
|
/// Cursor page of public local thoughts, newest-first, before `before`.
|
||||||
|
/// Used for OrderedCollectionPage responses.
|
||||||
|
async fn outbox_page_for_actor(
|
||||||
|
&self,
|
||||||
|
user_id: &UserId,
|
||||||
|
before: Option<chrono::DateTime<chrono::Utc>>,
|
||||||
|
limit: usize,
|
||||||
|
) -> Result<Vec<OutboxEntry>, DomainError>;
|
||||||
|
|
||||||
|
// ── Remote actor resolution ──────────────────────────────────────
|
||||||
|
|
||||||
|
/// Find the local UserId for a remote actor by its AP URL.
|
||||||
|
async fn find_remote_actor_id(
|
||||||
|
&self,
|
||||||
|
actor_ap_url: &url::Url,
|
||||||
|
) -> Result<Option<UserId>, DomainError>;
|
||||||
|
|
||||||
|
/// Ensure a remote actor placeholder exists; create one if absent.
|
||||||
|
/// Idempotent — safe to call multiple times with the same URL.
|
||||||
|
async fn intern_remote_actor(
|
||||||
|
&self,
|
||||||
|
actor_ap_url: &url::Url,
|
||||||
|
) -> Result<UserId, DomainError>;
|
||||||
|
|
||||||
|
// ── Inbox processing (remote → local) ───────────────────────────
|
||||||
|
|
||||||
|
/// Persist an incoming remote Note. Idempotent on ap_id.
|
||||||
|
async fn accept_note(
|
||||||
|
&self,
|
||||||
|
ap_id: &url::Url,
|
||||||
|
author_id: &UserId,
|
||||||
|
content: &str,
|
||||||
|
published: chrono::DateTime<chrono::Utc>,
|
||||||
|
sensitive: bool,
|
||||||
|
content_warning: Option<String>,
|
||||||
|
) -> Result<(), DomainError>;
|
||||||
|
|
||||||
|
/// Apply an Update to a previously accepted remote Note.
|
||||||
|
async fn apply_note_update(
|
||||||
|
&self,
|
||||||
|
ap_id: &url::Url,
|
||||||
|
new_content: &str,
|
||||||
|
) -> Result<(), DomainError>;
|
||||||
|
|
||||||
|
/// Remove a specific remote Note (Delete activity). Only touches
|
||||||
|
/// remotely-originated thoughts.
|
||||||
|
async fn retract_note(&self, ap_id: &url::Url) -> Result<(), DomainError>;
|
||||||
|
|
||||||
|
/// Remove all Notes from a remote actor (actor-level Delete/Tombstone).
|
||||||
|
async fn retract_actor_notes(
|
||||||
|
&self,
|
||||||
|
actor_ap_url: &url::Url,
|
||||||
|
) -> Result<(), DomainError>;
|
||||||
|
|
||||||
|
// ── Node-level stats ─────────────────────────────────────────────
|
||||||
|
|
||||||
|
/// Total locally-authored thought count for NodeInfo responses.
|
||||||
|
async fn count_local_notes(&self) -> Result<u64, DomainError>;
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
use std::sync::{Arc, Mutex};
|
use std::sync::{Arc, Mutex};
|
||||||
use async_trait::async_trait;
|
use async_trait::async_trait;
|
||||||
use chrono::Utc;
|
use chrono::Utc;
|
||||||
|
use url;
|
||||||
use crate::{
|
use crate::{
|
||||||
errors::DomainError,
|
errors::DomainError,
|
||||||
events::DomainEvent,
|
events::DomainEvent,
|
||||||
@@ -16,7 +17,7 @@ use crate::{
|
|||||||
user::User,
|
user::User,
|
||||||
},
|
},
|
||||||
ports::*,
|
ports::*,
|
||||||
value_objects::{ApiKeyId, Content, Email, NotificationId, ThoughtId, UserId, Username},
|
value_objects::{ApiKeyId, Content, Email, NotificationId, PasswordHash, ThoughtId, UserId, Username},
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Default, Clone)]
|
#[derive(Default, Clone)]
|
||||||
@@ -291,6 +292,48 @@ pub struct TestStore {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[async_trait] impl ActivityPubRepository for TestStore {
|
||||||
|
async fn outbox_entries_for_actor(&self, _uid: &UserId) -> Result<Vec<crate::ports::OutboxEntry>, DomainError> {
|
||||||
|
Ok(vec![])
|
||||||
|
}
|
||||||
|
async fn outbox_page_for_actor(&self, _uid: &UserId, _before: Option<chrono::DateTime<chrono::Utc>>, _limit: usize) -> Result<Vec<crate::ports::OutboxEntry>, DomainError> {
|
||||||
|
Ok(vec![])
|
||||||
|
}
|
||||||
|
async fn find_remote_actor_id(&self, actor_ap_url: &url::Url) -> Result<Option<UserId>, DomainError> {
|
||||||
|
let url = actor_ap_url.to_string();
|
||||||
|
Ok(self.users.lock().unwrap().iter()
|
||||||
|
.find(|u| u.ap_id.as_deref() == Some(&url))
|
||||||
|
.map(|u| u.id.clone()))
|
||||||
|
}
|
||||||
|
async fn intern_remote_actor(&self, actor_ap_url: &url::Url) -> Result<UserId, DomainError> {
|
||||||
|
if let Some(uid) = self.find_remote_actor_id(actor_ap_url).await? {
|
||||||
|
return Ok(uid);
|
||||||
|
}
|
||||||
|
let uid = UserId::new();
|
||||||
|
let handle = actor_ap_url.path().trim_start_matches('/').replace('/', "_");
|
||||||
|
let user = crate::models::user::User {
|
||||||
|
id: uid.clone(),
|
||||||
|
username: Username::from_trusted(handle.clone()),
|
||||||
|
email: Email::from_trusted(format!("{}@remote", uid)),
|
||||||
|
password_hash: PasswordHash("".into()),
|
||||||
|
display_name: None, bio: None, avatar_url: None, header_url: None,
|
||||||
|
custom_css: None, local: false,
|
||||||
|
ap_id: Some(actor_ap_url.to_string()),
|
||||||
|
inbox_url: None, public_key: None, private_key: None,
|
||||||
|
created_at: chrono::Utc::now(), updated_at: chrono::Utc::now(),
|
||||||
|
};
|
||||||
|
self.users.lock().unwrap().push(user);
|
||||||
|
Ok(uid)
|
||||||
|
}
|
||||||
|
async fn accept_note(&self, _ap_id: &url::Url, _author_id: &UserId, _content: &str, _published: chrono::DateTime<chrono::Utc>, _sensitive: bool, _content_warning: Option<String>) -> Result<(), DomainError> { Ok(()) }
|
||||||
|
async fn apply_note_update(&self, _ap_id: &url::Url, _new_content: &str) -> Result<(), DomainError> { Ok(()) }
|
||||||
|
async fn retract_note(&self, _ap_id: &url::Url) -> Result<(), DomainError> { Ok(()) }
|
||||||
|
async fn retract_actor_notes(&self, _actor_ap_url: &url::Url) -> Result<(), DomainError> { Ok(()) }
|
||||||
|
async fn count_local_notes(&self) -> Result<u64, DomainError> {
|
||||||
|
Ok(self.thoughts.lock().unwrap().iter().filter(|t| t.local).count() as u64)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[async_trait] impl EventPublisher for TestStore {
|
#[async_trait] impl EventPublisher for TestStore {
|
||||||
async fn publish(&self, event: &DomainEvent) -> Result<(), DomainError> {
|
async fn publish(&self, event: &DomainEvent) -> Result<(), DomainError> {
|
||||||
self.events.lock().unwrap().push(event.clone());
|
self.events.lock().unwrap().push(event.clone());
|
||||||
@@ -303,6 +346,28 @@ pub struct NoOpEventPublisher;
|
|||||||
async fn publish(&self, _e: &DomainEvent) -> Result<(), DomainError> { Ok(()) }
|
async fn publish(&self, _e: &DomainEvent) -> Result<(), DomainError> { Ok(()) }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod ap_repo_tests {
|
||||||
|
use super::*;
|
||||||
|
use crate::value_objects::UserId;
|
||||||
|
|
||||||
|
#[tokio::test]
|
||||||
|
async fn test_store_outbox_returns_empty() {
|
||||||
|
let store = TestStore::default();
|
||||||
|
let result = store.outbox_entries_for_actor(&UserId::new()).await.unwrap();
|
||||||
|
assert!(result.is_empty());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[tokio::test]
|
||||||
|
async fn test_store_intern_creates_placeholder() {
|
||||||
|
let store = TestStore::default();
|
||||||
|
let url = url::Url::parse("https://example.com/users/alice").unwrap();
|
||||||
|
let id1 = store.intern_remote_actor(&url).await.unwrap();
|
||||||
|
let id2 = store.intern_remote_actor(&url).await.unwrap();
|
||||||
|
assert_eq!(id1, id2, "intern must be idempotent");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod search_tests {
|
mod search_tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
|
|||||||
Reference in New Issue
Block a user