mirror of
https://github.com/Nutomic/ibis.git
synced 2024-11-22 17:11:09 +00:00
federate article create/update
This commit is contained in:
parent
5c1e753761
commit
f84c4f47f2
8 changed files with 232 additions and 30 deletions
46
src/api.rs
46
src/api.rs
|
@ -1,6 +1,9 @@
|
||||||
use crate::database::DatabaseHandle;
|
use crate::database::DatabaseHandle;
|
||||||
|
|
||||||
use crate::error::MyResult;
|
use crate::error::MyResult;
|
||||||
|
use crate::federation::activities::create_or_update_article::{
|
||||||
|
CreateOrUpdateArticle, CreateOrUpdateType,
|
||||||
|
};
|
||||||
use crate::federation::objects::article::DbArticle;
|
use crate::federation::objects::article::DbArticle;
|
||||||
use crate::federation::objects::instance::DbInstance;
|
use crate::federation::objects::instance::DbInstance;
|
||||||
use crate::utils::generate_object_id;
|
use crate::utils::generate_object_id;
|
||||||
|
@ -16,7 +19,10 @@ use url::Url;
|
||||||
|
|
||||||
pub fn api_routes() -> Router {
|
pub fn api_routes() -> Router {
|
||||||
Router::new()
|
Router::new()
|
||||||
.route("/article", get(get_article).post(create_article))
|
.route(
|
||||||
|
"/article",
|
||||||
|
get(get_article).post(create_article).patch(edit_article),
|
||||||
|
)
|
||||||
.route("/resolve_object", get(resolve_object))
|
.route("/resolve_object", get(resolve_object))
|
||||||
.route("/instance", get(get_local_instance))
|
.route("/instance", get(get_local_instance))
|
||||||
.route("/instance/follow", post(follow_instance))
|
.route("/instance/follow", post(follow_instance))
|
||||||
|
@ -42,8 +48,46 @@ async fn create_article(
|
||||||
instance: local_instance_id,
|
instance: local_instance_id,
|
||||||
local: true,
|
local: true,
|
||||||
};
|
};
|
||||||
|
{
|
||||||
let mut articles = data.articles.lock().unwrap();
|
let mut articles = data.articles.lock().unwrap();
|
||||||
articles.insert(article.ap_id.inner().clone(), article.clone());
|
articles.insert(article.ap_id.inner().clone(), article.clone());
|
||||||
|
}
|
||||||
|
|
||||||
|
CreateOrUpdateArticle::send_to_local_followers(
|
||||||
|
article.clone(),
|
||||||
|
CreateOrUpdateType::Create,
|
||||||
|
&data,
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
Ok(Json(article))
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Deserialize, Serialize)]
|
||||||
|
pub struct EditArticle {
|
||||||
|
pub ap_id: ObjectId<DbArticle>,
|
||||||
|
pub new_text: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[debug_handler]
|
||||||
|
async fn edit_article(
|
||||||
|
data: Data<DatabaseHandle>,
|
||||||
|
Form(edit_article): Form<EditArticle>,
|
||||||
|
) -> MyResult<Json<DbArticle>> {
|
||||||
|
let article = {
|
||||||
|
let mut lock = data.articles.lock().unwrap();
|
||||||
|
let article = lock.get_mut(edit_article.ap_id.inner()).unwrap();
|
||||||
|
article.text = edit_article.new_text;
|
||||||
|
article.clone()
|
||||||
|
};
|
||||||
|
|
||||||
|
CreateOrUpdateArticle::send_to_local_followers(
|
||||||
|
article.clone(),
|
||||||
|
CreateOrUpdateType::Update,
|
||||||
|
&data,
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
|
||||||
Ok(Json(article))
|
Ok(Json(article))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
85
src/federation/activities/create_or_update_article.rs
Normal file
85
src/federation/activities/create_or_update_article.rs
Normal file
|
@ -0,0 +1,85 @@
|
||||||
|
use crate::database::DatabaseHandle;
|
||||||
|
use crate::error::MyResult;
|
||||||
|
use crate::federation::objects::article::{Article, DbArticle};
|
||||||
|
use crate::federation::objects::instance::DbInstance;
|
||||||
|
use crate::utils::generate_object_id;
|
||||||
|
use activitypub_federation::{
|
||||||
|
config::Data,
|
||||||
|
fetch::object_id::ObjectId,
|
||||||
|
protocol::helpers::deserialize_one_or_many,
|
||||||
|
traits::{ActivityHandler, Object},
|
||||||
|
};
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
use url::Url;
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, Deserialize, Serialize, PartialEq, Eq)]
|
||||||
|
pub enum CreateOrUpdateType {
|
||||||
|
Create,
|
||||||
|
Update,
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: temporary placeholder, later rework this to send diffs
|
||||||
|
#[derive(Deserialize, Serialize, Debug)]
|
||||||
|
#[serde(rename_all = "camelCase")]
|
||||||
|
pub struct CreateOrUpdateArticle {
|
||||||
|
pub actor: ObjectId<DbInstance>,
|
||||||
|
#[serde(deserialize_with = "deserialize_one_or_many")]
|
||||||
|
pub to: Vec<Url>,
|
||||||
|
pub object: Article,
|
||||||
|
#[serde(rename = "type")]
|
||||||
|
pub kind: CreateOrUpdateType,
|
||||||
|
pub id: Url,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl CreateOrUpdateArticle {
|
||||||
|
pub async fn send_to_local_followers(
|
||||||
|
article: DbArticle,
|
||||||
|
kind: CreateOrUpdateType,
|
||||||
|
data: &Data<DatabaseHandle>,
|
||||||
|
) -> MyResult<()> {
|
||||||
|
let local_instance = data.local_instance();
|
||||||
|
let to = local_instance
|
||||||
|
.followers
|
||||||
|
.iter()
|
||||||
|
.map(|f| f.ap_id.inner().clone())
|
||||||
|
.collect();
|
||||||
|
let object = article.clone().into_json(data).await?;
|
||||||
|
let id = generate_object_id(local_instance.ap_id.inner())?;
|
||||||
|
let create_or_update = CreateOrUpdateArticle {
|
||||||
|
actor: local_instance.ap_id.clone(),
|
||||||
|
to,
|
||||||
|
object,
|
||||||
|
kind,
|
||||||
|
id,
|
||||||
|
};
|
||||||
|
let inboxes = local_instance
|
||||||
|
.followers
|
||||||
|
.iter()
|
||||||
|
.map(|f| f.inbox.clone())
|
||||||
|
.collect();
|
||||||
|
local_instance.send(create_or_update, inboxes, data).await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#[async_trait::async_trait]
|
||||||
|
impl ActivityHandler for CreateOrUpdateArticle {
|
||||||
|
type DataType = DatabaseHandle;
|
||||||
|
type Error = crate::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> {
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn receive(self, data: &Data<Self::DataType>) -> Result<(), Self::Error> {
|
||||||
|
DbArticle::from_json(self.object, data).await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
|
@ -50,15 +50,14 @@ impl ActivityHandler for Follow {
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn receive(self, data: &Data<Self::DataType>) -> Result<(), Self::Error> {
|
async fn receive(self, data: &Data<Self::DataType>) -> Result<(), Self::Error> {
|
||||||
dbg!(&self);
|
let actor = self.actor.dereference(data).await?;
|
||||||
// add to followers
|
// add to followers
|
||||||
let local_instance = {
|
let local_instance = {
|
||||||
let mut lock = data.instances.lock().unwrap();
|
let mut lock = data.instances.lock().unwrap();
|
||||||
let local_instance = lock.iter_mut().find(|i| i.1.local).unwrap().1;
|
let local_instance = lock.iter_mut().find(|i| i.1.local).unwrap().1;
|
||||||
local_instance.followers.push(self.actor.inner().clone());
|
local_instance.followers.push(actor);
|
||||||
local_instance.clone()
|
local_instance.clone()
|
||||||
};
|
};
|
||||||
dbg!(&local_instance.followers.len());
|
|
||||||
|
|
||||||
// send back an accept
|
// send back an accept
|
||||||
let follower = self.actor.dereference(data).await?;
|
let follower = self.actor.dereference(data).await?;
|
||||||
|
|
|
@ -1,2 +1,3 @@
|
||||||
pub mod accept;
|
pub mod accept;
|
||||||
|
pub mod create_or_update_article;
|
||||||
pub mod follow;
|
pub mod follow;
|
||||||
|
|
|
@ -23,7 +23,7 @@ pub struct DbInstance {
|
||||||
pub(crate) public_key: String,
|
pub(crate) public_key: String,
|
||||||
pub(crate) private_key: Option<String>,
|
pub(crate) private_key: Option<String>,
|
||||||
pub(crate) last_refreshed_at: DateTime<Utc>,
|
pub(crate) last_refreshed_at: DateTime<Utc>,
|
||||||
pub followers: Vec<Url>,
|
pub followers: Vec<DbInstance>,
|
||||||
pub follows: Vec<Url>,
|
pub follows: Vec<Url>,
|
||||||
pub local: bool,
|
pub local: bool,
|
||||||
}
|
}
|
||||||
|
@ -40,10 +40,6 @@ pub struct Instance {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DbInstance {
|
impl DbInstance {
|
||||||
pub fn followers(&self) -> &Vec<Url> {
|
|
||||||
&self.followers
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn followers_url(&self) -> Result<Url, Error> {
|
pub fn followers_url(&self) -> Result<Url, Error> {
|
||||||
Ok(Url::parse(&format!("{}/followers", self.ap_id.inner()))?)
|
Ok(Url::parse(&format!("{}/followers", self.ap_id.inner()))?)
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,6 +11,7 @@ use activitypub_federation::protocol::context::WithContext;
|
||||||
use activitypub_federation::traits::Object;
|
use activitypub_federation::traits::Object;
|
||||||
use activitypub_federation::traits::{ActivityHandler, Collection};
|
use activitypub_federation::traits::{ActivityHandler, Collection};
|
||||||
|
|
||||||
|
use crate::federation::activities::create_or_update_article::CreateOrUpdateArticle;
|
||||||
use crate::federation::objects::articles_collection::{ArticleCollection, DbArticleCollection};
|
use crate::federation::objects::articles_collection::{ArticleCollection, DbArticleCollection};
|
||||||
use axum::response::IntoResponse;
|
use axum::response::IntoResponse;
|
||||||
use axum::routing::{get, post};
|
use axum::routing::{get, post};
|
||||||
|
@ -50,6 +51,7 @@ async fn http_get_articles(
|
||||||
pub enum InboxActivities {
|
pub enum InboxActivities {
|
||||||
Follow(Follow),
|
Follow(Follow),
|
||||||
Accept(Accept),
|
Accept(Accept),
|
||||||
|
CreateOrUpdateArticle(CreateOrUpdateArticle),
|
||||||
}
|
}
|
||||||
|
|
||||||
#[debug_handler]
|
#[debug_handler]
|
||||||
|
|
|
@ -1,10 +1,13 @@
|
||||||
|
use fediwiki::api::{FollowInstance, ResolveObject};
|
||||||
use fediwiki::error::MyResult;
|
use fediwiki::error::MyResult;
|
||||||
|
use fediwiki::federation::objects::instance::DbInstance;
|
||||||
use once_cell::sync::Lazy;
|
use once_cell::sync::Lazy;
|
||||||
use reqwest::Client;
|
use reqwest::Client;
|
||||||
use serde::de::Deserialize;
|
use serde::de::Deserialize;
|
||||||
use serde::ser::Serialize;
|
use serde::ser::Serialize;
|
||||||
use std::sync::Once;
|
use std::sync::Once;
|
||||||
use tracing::log::LevelFilter;
|
use tracing::log::LevelFilter;
|
||||||
|
use url::Url;
|
||||||
|
|
||||||
pub static CLIENT: Lazy<Client> = Lazy::new(Client::new);
|
pub static CLIENT: Lazy<Client> = Lazy::new(Client::new);
|
||||||
|
|
||||||
|
@ -51,3 +54,37 @@ where
|
||||||
.json()
|
.json()
|
||||||
.await?)
|
.await?)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub async fn patch<T: Serialize, R>(hostname: &str, endpoint: &str, form: &T) -> MyResult<R>
|
||||||
|
where
|
||||||
|
R: for<'de> Deserialize<'de>,
|
||||||
|
{
|
||||||
|
Ok(CLIENT
|
||||||
|
.patch(format!("http://{}/api/v1/{}", hostname, endpoint))
|
||||||
|
.form(form)
|
||||||
|
.send()
|
||||||
|
.await?
|
||||||
|
.json()
|
||||||
|
.await?)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn follow_instance(follow_instance: &str, followed_instance: &str) -> MyResult<()> {
|
||||||
|
// fetch beta instance on alpha
|
||||||
|
let resolve_form = ResolveObject {
|
||||||
|
id: Url::parse(&format!("http://{}", followed_instance))?,
|
||||||
|
};
|
||||||
|
let beta_instance_resolved: DbInstance =
|
||||||
|
get_query(followed_instance, "resolve_object", Some(resolve_form)).await?;
|
||||||
|
|
||||||
|
// send follow
|
||||||
|
let follow_form = FollowInstance {
|
||||||
|
instance_id: beta_instance_resolved.ap_id,
|
||||||
|
};
|
||||||
|
// cant use post helper because follow doesnt return json
|
||||||
|
CLIENT
|
||||||
|
.post(format!("http://{}/api/v1/instance/follow", follow_instance))
|
||||||
|
.form(&follow_form)
|
||||||
|
.send()
|
||||||
|
.await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
|
@ -2,9 +2,9 @@ extern crate fediwiki;
|
||||||
|
|
||||||
mod common;
|
mod common;
|
||||||
|
|
||||||
use crate::common::{get_query, post, setup, CLIENT};
|
use crate::common::{follow_instance, get_query, patch, post, setup};
|
||||||
use common::get;
|
use common::get;
|
||||||
use fediwiki::api::{CreateArticle, FollowInstance, GetArticle, ResolveObject};
|
use fediwiki::api::{CreateArticle, EditArticle, GetArticle, ResolveObject};
|
||||||
use fediwiki::error::MyResult;
|
use fediwiki::error::MyResult;
|
||||||
use fediwiki::federation::objects::article::DbArticle;
|
use fediwiki::federation::objects::article::DbArticle;
|
||||||
use fediwiki::federation::objects::instance::DbInstance;
|
use fediwiki::federation::objects::instance::DbInstance;
|
||||||
|
@ -69,23 +69,7 @@ async fn test_follow_instance() -> MyResult<()> {
|
||||||
let beta_instance: DbInstance = get(hostname_beta, "instance").await?;
|
let beta_instance: DbInstance = get(hostname_beta, "instance").await?;
|
||||||
assert_eq!(0, beta_instance.followers.len());
|
assert_eq!(0, beta_instance.followers.len());
|
||||||
|
|
||||||
// fetch beta instance on alpha
|
common::follow_instance(hostname_alpha, &hostname_beta).await?;
|
||||||
let resolve_object = ResolveObject {
|
|
||||||
id: Url::parse(&format!("http://{hostname_beta}"))?,
|
|
||||||
};
|
|
||||||
let beta_instance_resolved: DbInstance =
|
|
||||||
get_query(hostname_beta, "resolve_object", Some(resolve_object)).await?;
|
|
||||||
|
|
||||||
// send follow
|
|
||||||
let follow_instance = FollowInstance {
|
|
||||||
instance_id: beta_instance_resolved.ap_id,
|
|
||||||
};
|
|
||||||
// cant use post helper because follow doesnt return json
|
|
||||||
CLIENT
|
|
||||||
.post(format!("http://{hostname_alpha}/api/v1/instance/follow"))
|
|
||||||
.form(&follow_instance)
|
|
||||||
.send()
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
// check that follow was federated
|
// check that follow was federated
|
||||||
let beta_instance: DbInstance = get(hostname_beta, "instance").await?;
|
let beta_instance: DbInstance = get(hostname_beta, "instance").await?;
|
||||||
|
@ -155,3 +139,57 @@ async fn test_synchronize_articles() -> MyResult<()> {
|
||||||
handle_beta.abort();
|
handle_beta.abort();
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[tokio::test]
|
||||||
|
#[serial]
|
||||||
|
async fn test_federate_article_changes() -> MyResult<()> {
|
||||||
|
setup();
|
||||||
|
let hostname_alpha = "localhost:8131";
|
||||||
|
let hostname_beta = "localhost:8132";
|
||||||
|
let handle_alpha = tokio::task::spawn(async {
|
||||||
|
start(hostname_alpha).await.unwrap();
|
||||||
|
});
|
||||||
|
let handle_beta = tokio::task::spawn(async {
|
||||||
|
start(hostname_beta).await.unwrap();
|
||||||
|
});
|
||||||
|
|
||||||
|
follow_instance(hostname_alpha, hostname_beta).await?;
|
||||||
|
|
||||||
|
// create new article
|
||||||
|
let create_form = CreateArticle {
|
||||||
|
title: "Manu_Chao".to_string(),
|
||||||
|
text: "Lorem ipsum".to_string(),
|
||||||
|
};
|
||||||
|
let create_res: DbArticle = post(hostname_beta, "article", &create_form).await?;
|
||||||
|
assert_eq!(create_res.title, create_form.title);
|
||||||
|
|
||||||
|
// article should be federated to alpha
|
||||||
|
let get_article = GetArticle {
|
||||||
|
title: create_res.title.clone(),
|
||||||
|
};
|
||||||
|
let get_res =
|
||||||
|
get_query::<DbArticle, _>(hostname_alpha, "article", Some(get_article.clone())).await?;
|
||||||
|
assert_eq!(create_res.title, get_res.title);
|
||||||
|
assert_eq!(create_res.text, get_res.text);
|
||||||
|
|
||||||
|
// edit the article
|
||||||
|
let edit_form = EditArticle {
|
||||||
|
ap_id: create_res.ap_id,
|
||||||
|
new_text: "Lorem Ipsum 2".to_string(),
|
||||||
|
};
|
||||||
|
let edit_res: DbArticle = patch(hostname_beta, "article", &edit_form).await?;
|
||||||
|
assert_eq!(edit_res.text, edit_form.new_text);
|
||||||
|
|
||||||
|
// edit should be federated to alpha
|
||||||
|
let get_article = GetArticle {
|
||||||
|
title: edit_res.title.clone(),
|
||||||
|
};
|
||||||
|
let get_res =
|
||||||
|
get_query::<DbArticle, _>(hostname_alpha, "article", Some(get_article.clone())).await?;
|
||||||
|
assert_eq!(edit_res.title, get_res.title);
|
||||||
|
assert_eq!(edit_res.text, get_res.text);
|
||||||
|
|
||||||
|
handle_alpha.abort();
|
||||||
|
handle_beta.abort();
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue