1
0
Fork 0
forked from alemi/upub

chore: cleaned up all model modules

This commit is contained in:
əlemi 2024-03-23 06:32:15 +01:00
parent 1a7a4f6df6
commit 72c2cd5f81
Signed by: alemi
GPG key ID: A4895B84D311642C
7 changed files with 103 additions and 319 deletions

View file

@ -1,9 +1,22 @@
use axum::{extract::{Path, State}, http::StatusCode}; use axum::{extract::{Path, State}, http::StatusCode};
use sea_orm::EntityTrait; use sea_orm::EntityTrait;
use crate::{activitystream::{object::activity::ActivityMut, Base, Node}, model::{activity, object}, server::Context}; use crate::{activitystream::{object::{activity::ActivityMut, ObjectMut}, BaseMut, Node}, model::{self, activity, object}, server::Context};
use super::{jsonld::LD, JsonLD}; use super::{jsonld::LD, JsonLD};
pub fn ap_activity(activity: model::activity::Model) -> serde_json::Value {
serde_json::Value::new_object()
.set_id(Some(&activity.id))
.set_activity_type(Some(activity.activity_type))
.set_actor(Node::link(activity.actor))
.set_object(Node::maybe_link(activity.object))
.set_target(Node::maybe_link(activity.target))
.set_published(Some(activity.published))
.set_to(Node::links(activity.to.0.clone()))
.set_bto(Node::empty())
.set_cc(Node::links(activity.cc.0.clone()))
.set_bcc(Node::empty())
}
pub async fn view(State(ctx) : State<Context>, Path(id): Path<String>) -> Result<JsonLD<serde_json::Value>, StatusCode> { pub async fn view(State(ctx) : State<Context>, Path(id): Path<String>) -> Result<JsonLD<serde_json::Value>, StatusCode> {
match activity::Entity::find_by_id(ctx.aid(id)) match activity::Entity::find_by_id(ctx.aid(id))
@ -12,9 +25,8 @@ pub async fn view(State(ctx) : State<Context>, Path(id): Path<String>) -> Result
.await .await
{ {
Ok(Some((activity, object))) => Ok(JsonLD( Ok(Some((activity, object))) => Ok(JsonLD(
activity ap_activity(activity)
.underlying_json_object() .set_object(Node::maybe_object(object.map(super::object::ap_object)))
.set_object(Node::maybe_object(object))
.ld_context() .ld_context()
)), )),
Ok(None) => Err(StatusCode::NOT_FOUND), Ok(None) => Err(StatusCode::NOT_FOUND),

View file

@ -1,14 +1,29 @@
use axum::{extract::{Path, State}, http::StatusCode}; use axum::{extract::{Path, State}, http::StatusCode};
use sea_orm::EntityTrait; use sea_orm::EntityTrait;
use crate::{activitystream::Base, model::object, server::Context}; use crate::{activitystream::{object::ObjectMut, BaseMut, Node}, model::{self, object}, server::Context};
use super::{jsonld::LD, JsonLD}; use super::{jsonld::LD, JsonLD};
pub fn ap_object(object: model::object::Model) -> serde_json::Value {
serde_json::Value::new_object()
.set_id(Some(&object.id))
.set_object_type(Some(object.object_type))
.set_attributed_to(Node::maybe_link(object.attributed_to))
.set_name(object.name.as_deref())
.set_summary(object.summary.as_deref())
.set_content(object.content.as_deref())
.set_context(Node::maybe_link(object.context.clone()))
.set_published(Some(object.published))
.set_to(Node::links(object.to.0.clone()))
.set_bto(Node::empty())
.set_cc(Node::links(object.cc.0.clone()))
.set_bcc(Node::empty())
}
pub async fn view(State(ctx) : State<Context>, Path(id): Path<String>) -> Result<JsonLD<serde_json::Value>, StatusCode> { pub async fn view(State(ctx) : State<Context>, Path(id): Path<String>) -> Result<JsonLD<serde_json::Value>, StatusCode> {
match object::Entity::find_by_id(ctx.oid(id)).one(ctx.db()).await { match object::Entity::find_by_id(ctx.oid(id)).one(ctx.db()).await {
Ok(Some(object)) => Ok(JsonLD(object.underlying_json_object().ld_context())), Ok(Some(object)) => Ok(JsonLD(ap_object(object).ld_context())),
Ok(None) => Err(StatusCode::NOT_FOUND), Ok(None) => Err(StatusCode::NOT_FOUND),
Err(e) => { Err(e) => {
tracing::error!("error querying for object: {e}"); tracing::error!("error querying for object: {e}");

View file

@ -63,6 +63,7 @@ pub async fn view(State(ctx) : State<Context>, Path(id): Path<String>) -> Result
// .set_public_key(user.public_key) // TODO // .set_public_key(user.public_key) // TODO
.set_discoverable(Some(true)) .set_discoverable(Some(true))
.set_endpoints(None) .set_endpoints(None)
.ld_context()
)) ))
}, },
// remote user // remote user

View file

@ -1,7 +1,7 @@
use axum::{extract::{Path, Query, State}, http::StatusCode}; use axum::{extract::{Path, Query, State}, http::StatusCode};
use sea_orm::{EntityTrait, Order, QueryOrder, QuerySelect}; use sea_orm::{EntityTrait, Order, QueryOrder, QuerySelect};
use crate::{activitypub::{jsonld::LD, JsonLD, Pagination}, activitystream::{object::{activity::ActivityMut, collection::{page::CollectionPageMut, CollectionMut, CollectionType}}, Base, BaseMut, Node}, model::{activity, object}, server::Context, url}; use crate::{activitypub::{jsonld::LD, JsonLD, Pagination}, activitystream::{object::{activity::ActivityMut, collection::{page::CollectionPageMut, CollectionMut, CollectionType}}, BaseMut, Node}, model::{activity, object}, server::Context, url};
pub async fn outbox( pub async fn outbox(
State(ctx): State<Context>, State(ctx): State<Context>,
@ -23,7 +23,10 @@ pub async fn outbox(
let next = ctx.id(items.last().map(|(a, _o)| a.id.as_str()).unwrap_or("").to_string()); let next = ctx.id(items.last().map(|(a, _o)| a.id.as_str()).unwrap_or("").to_string());
let items = items let items = items
.into_iter() .into_iter()
.map(|(a, o)| a.underlying_json_object().set_object(Node::maybe_object(o))) .map(|(a, o)|
super::super::activity::ap_activity(a)
.set_object(Node::maybe_object(o.map(super::super::object::ap_object)))
)
.collect(); .collect();
Ok(JsonLD( Ok(JsonLD(
serde_json::Value::new_object() serde_json::Value::new_object()

View file

@ -1,6 +1,6 @@
use sea_orm::entity::prelude::*; use sea_orm::entity::prelude::*;
use crate::{activitypub::jsonld::LD, activitystream::{link::Link, object::{activity::{Activity, ActivityMut, ActivityType}, actor::Actor, Object, ObjectMut, ObjectType}, Base, BaseMut, BaseType, Node}}; use crate::activitystream::object::activity::{Activity, ActivityType};
use super::Audience; use super::Audience;
@ -25,6 +25,23 @@ pub struct Model {
// TODO: origin, result, instrument // TODO: origin, result, instrument
} }
impl Model {
pub fn new(activity: &impl Activity) -> Result<Self, super::FieldError> {
Ok(Model {
id: activity.id().ok_or(super::FieldError("id"))?.to_string(),
activity_type: activity.activity_type().ok_or(super::FieldError("type"))?,
actor: activity.actor().id().ok_or(super::FieldError("actor"))?.to_string(),
object: activity.object().id().map(|x| x.to_string()),
target: activity.target().id().map(|x| x.to_string()),
published: activity.published().unwrap_or(chrono::Utc::now()),
to: activity.to().into(),
bto: activity.bto().into(),
cc: activity.cc().into(),
bcc: activity.bcc().into(),
})
}
}
#[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)] #[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)]
pub enum Relation { pub enum Relation {
#[sea_orm( #[sea_orm(
@ -55,94 +72,3 @@ impl Related<super::object::Entity> for Entity {
} }
impl ActiveModelBehavior for ActiveModel {} impl ActiveModelBehavior for ActiveModel {}
impl Base for Model {
fn id(&self) -> Option<&str> {
Some(&self.id)
}
fn base_type(&self) -> Option<BaseType> {
Some(BaseType::Object(ObjectType::Activity(self.activity_type)))
}
fn underlying_json_object(self) -> serde_json::Value {
serde_json::Value::new_object()
.set_id(Some(&self.id))
.set_activity_type(Some(self.activity_type))
.set_actor(Node::link(self.actor))
.set_object(Node::maybe_link(self.object))
.set_target(Node::maybe_link(self.target))
.set_published(Some(self.published))
.set_to(Node::links(self.to.0.clone()))
.set_bto(Node::empty())
.set_cc(Node::links(self.cc.0.clone()))
.set_bcc(Node::empty())
}
}
impl Object for Model {
fn object_type(&self) -> Option<ObjectType> {
Some(ObjectType::Activity(self.activity_type))
}
fn published(&self) -> Option<chrono::DateTime<chrono::Utc>> {
Some(self.published)
}
fn to(&self) -> Node<impl Link> {
Node::links(self.to.0.clone())
}
fn bto(&self) -> Node<impl Link> {
Node::links(self.bto.0.clone())
}
fn cc(&self) -> Node<impl Link> {
Node::links(self.cc.0.clone())
}
fn bcc(&self) -> Node<impl Link> {
Node::links(self.bcc.0.clone())
}
}
impl Activity for Model {
fn activity_type(&self) -> Option<ActivityType> {
Some(self.activity_type)
}
fn actor(&self) -> Node<impl Actor> {
Node::<serde_json::Value>::Link(Box::new(self.actor.clone()))
}
fn object(&self) -> Node<impl Object> {
match &self.object {
None => Node::Empty::<serde_json::Value>,
Some(x) => Node::Link(Box::new(x.clone())),
}
}
fn target(&self) -> Node<impl Object> {
match &self.target {
None => Node::Empty::<serde_json::Value>,
Some(x) => Node::Link(Box::new(x.clone())),
}
}
}
impl Model {
pub fn new(activity: &impl Activity) -> Result<Self, super::FieldError> {
Ok(Model {
id: activity.id().ok_or(super::FieldError("id"))?.to_string(),
activity_type: activity.activity_type().ok_or(super::FieldError("type"))?,
actor: activity.actor().id().ok_or(super::FieldError("actor"))?.to_string(),
object: activity.object().id().map(|x| x.to_string()),
target: activity.target().id().map(|x| x.to_string()),
published: activity.published().unwrap_or(chrono::Utc::now()),
to: activity.to().into(),
bto: activity.bto().into(),
cc: activity.cc().into(),
bcc: activity.bcc().into(),
})
}
}

View file

@ -1,6 +1,6 @@
use sea_orm::entity::prelude::*; use sea_orm::entity::prelude::*;
use crate::{activitypub::jsonld::LD, activitystream::{object::{ObjectMut, ObjectType}, BaseMut, BaseType, Link, Node}}; use crate::activitystream::object::ObjectType;
use super::Audience; use super::Audience;
@ -26,6 +26,28 @@ pub struct Model {
pub published: ChronoDateTimeUtc, pub published: ChronoDateTimeUtc,
} }
impl Model {
pub fn new(object: &impl crate::activitystream::object::Object) -> Result<Self, super::FieldError> {
Ok(Model {
id: object.id().ok_or(super::FieldError("id"))?.to_string(),
object_type: object.object_type().ok_or(super::FieldError("type"))?,
attributed_to: object.attributed_to().id().map(|x| x.to_string()),
name: object.name().map(|x| x.to_string()),
summary: object.summary().map(|x| x.to_string()),
content: object.content().map(|x| x.to_string()),
context: object.context().id().map(|x| x.to_string()),
published: object.published().ok_or(super::FieldError("published"))?,
comments: 0,
likes: 0,
shares: 0,
to: object.to().into(),
bto: object.bto().into(),
cc: object.cc().into(),
bcc: object.bcc().into(),
})
}
}
#[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)] #[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)]
pub enum Relation { pub enum Relation {
#[sea_orm(has_many = "super::activity::Entity")] #[sea_orm(has_many = "super::activity::Entity")]
@ -52,97 +74,3 @@ impl Related<super::user::Entity> for Entity {
} }
impl ActiveModelBehavior for ActiveModel {} impl ActiveModelBehavior for ActiveModel {}
impl crate::activitystream::Base for Model {
fn id(&self) -> Option<&str> {
Some(&self.id)
}
fn base_type(&self) -> Option<BaseType> {
Some(BaseType::Object(self.object_type))
}
fn underlying_json_object(self) -> serde_json::Value {
serde_json::Value::new_object()
.set_id(Some(&self.id))
.set_object_type(Some(self.object_type))
.set_attributed_to(Node::maybe_link(self.attributed_to))
.set_name(self.name.as_deref())
.set_summary(self.summary.as_deref())
.set_content(self.content.as_deref())
.set_context(Node::maybe_link(self.context.clone()))
.set_published(Some(self.published))
.set_to(Node::links(self.to.0.clone()))
.set_bto(Node::empty())
.set_cc(Node::links(self.cc.0.clone()))
.set_bcc(Node::empty())
}
}
impl crate::activitystream::object::Object for Model {
fn object_type(&self) -> Option<ObjectType> {
Some(self.object_type)
}
fn attributed_to(&self) -> Node<impl crate::activitystream::object::actor::Actor> {
Node::<serde_json::Value>::from(self.attributed_to.as_deref())
}
fn name(&self) -> Option<&str> {
self.name.as_deref()
}
fn summary(&self) -> Option<&str> {
self.summary.as_deref()
}
fn content(&self) -> Option<&str> {
self.content.as_deref()
}
fn context(&self) -> Node<impl crate::activitystream::Object> {
Node::maybe_link(self.context.clone())
}
fn to(&self) -> Node<impl Link> {
Node::links(self.to.0.clone())
}
fn bto(&self) -> Node<impl Link> {
Node::links(self.bto.0.clone())
}
fn cc(&self) -> Node<impl Link> {
Node::links(self.cc.0.clone())
}
fn bcc(&self) -> Node<impl Link> {
Node::links(self.bcc.0.clone())
}
fn published (&self) -> Option<chrono::DateTime<chrono::Utc>> {
Some(self.published)
}
}
impl Model {
pub fn new(object: &impl crate::activitystream::object::Object) -> Result<Self, super::FieldError> {
Ok(Model {
id: object.id().ok_or(super::FieldError("id"))?.to_string(),
object_type: object.object_type().ok_or(super::FieldError("type"))?,
attributed_to: object.attributed_to().id().map(|x| x.to_string()),
name: object.name().map(|x| x.to_string()),
summary: object.summary().map(|x| x.to_string()),
content: object.content().map(|x| x.to_string()),
context: object.context().id().map(|x| x.to_string()),
published: object.published().ok_or(super::FieldError("published"))?,
comments: 0,
likes: 0,
shares: 0,
to: object.to().into(),
bto: object.bto().into(),
cc: object.cc().into(),
bcc: object.bcc().into(),
})
}
}

View file

@ -35,6 +35,29 @@ pub struct Model {
// pub streams: Option<String>, // pub streams: Option<String>,
} }
impl Model {
pub fn new(object: &impl Actor) -> Result<Self, super::FieldError> {
let ap_id = object.id().ok_or(super::FieldError("id"))?.to_string();
let (domain, preferred_username) = activitypub::split_id(&ap_id);
Ok(Model {
id: ap_id, preferred_username, domain,
actor_type: object.actor_type().ok_or(super::FieldError("type"))?,
name: object.name().map(|x| x.to_string()),
summary: object.summary().map(|x| x.to_string()),
icon: object.icon().id().map(|x| x.to_string()),
image: object.image().id().map(|x| x.to_string()),
inbox: object.inbox().id().map(|x| x.to_string()),
outbox: object.inbox().id().map(|x| x.to_string()),
shared_inbox: None, // TODO!!! parse endpoints
followers: object.followers().id().map(|x| x.to_string()),
following: object.following().id().map(|x| x.to_string()),
created: object.published().unwrap_or(chrono::Utc::now()),
updated: chrono::Utc::now(),
public_key: object.public_key().get().ok_or(super::FieldError("publicKey"))?.public_key_pem().to_string(),
private_key: None, // there's no way to transport privkey over AP json, must come from DB
})
}
}
#[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)] #[derive(Copy, Clone, Debug, EnumIter, DeriveRelation)]
pub enum Relation { pub enum Relation {
@ -76,127 +99,3 @@ impl Related<super::credential::Entity> for Entity {
} }
impl ActiveModelBehavior for ActiveModel {} impl ActiveModelBehavior for ActiveModel {}
// impl crate::activitystream::Base for Model {
// fn id(&self) -> Option<&str> {
// Some(&self.id)
// }
//
// fn base_type(&self) -> Option<BaseType> {
// Some(BaseType::Object(ObjectType::Actor(self.actor_type)))
// }
//
// fn underlying_json_object(self) -> serde_json::Value {
// serde_json::Value::new_object()
// .set_id(Some(&self.id))
// .set_actor_type(Some(self.actor_type))
// .set_name(self.name.as_deref())
// .set_summary(self.summary.as_deref())
// .set_icon(self.icon())
// .set_image(self.image())
// .set_published(Some(self.created))
// .set_preferred_username(Some(&self.preferred_username))
// .set_inbox(self.inbox())
// .set_outbox(self.outbox())
// .set_following(self.following())
// .set_followers(self.followers())
// .set_public_key(self.public_key())
// .set_discoverable(Some(true))
// .set_endpoints(None) // TODO dirty fix to put an empty object
// }
// }
//
// impl crate::activitystream::object::Object for Model {
// fn name(&self) -> Option<&str> {
// self.name.as_deref()
// }
//
// fn summary(&self) -> Option<&str> {
// self.summary.as_deref()
// }
//
// fn icon(&self) -> Node<impl Image> {
// match &self.icon {
// Some(x) => Node::object(
// serde_json::Value::new_object()
// .set_document_type(Some(DocumentType::Image))
// .set_url(Node::link(x.clone()))
// ),
// None => Node::Empty,
// }
// }
//
// fn image(&self) -> Node<impl Image> {
// match &self.image {
// Some(x) => Node::object(
// serde_json::Value::new_object()
// .set_document_type(Some(DocumentType::Image))
// .set_url(Node::link(x.clone()))
// ),
// None => Node::Empty,
// }
// }
//
// fn published(&self) -> Option<chrono::DateTime<chrono::Utc>> {
// Some(self.created)
// }
// }
//
// impl crate::activitystream::object::actor::Actor for Model {
// fn actor_type(&self) -> Option<ActorType> {
// Some(self.actor_type)
// }
//
// fn preferred_username(&self) -> Option<&str> {
// Some(&self.preferred_username)
// }
//
// fn inbox(&self) -> Node<impl Collection> {
// Node::link(self.inbox.clone().unwrap_or(format!("https://{}/users/{}/inbox", self.domain, self.preferred_username)))
// }
//
// fn outbox(&self) -> Node<impl Collection> {
// Node::link(self.outbox.clone().unwrap_or(format!("https://{}/users/{}/outbox", self.domain, self.preferred_username)))
// }
//
// fn following(&self) -> Node<impl Collection> {
// Node::link(self.following.clone().unwrap_or(format!("https://{}/users/{}/following", self.domain, self.preferred_username)))
// }
//
// fn followers(&self) -> Node<impl Collection> {
// Node::link(self.following.clone().unwrap_or(format!("https://{}/users/{}/followers", self.domain, self.preferred_username)))
// }
//
// fn public_key(&self) -> Node<impl crate::activitystream::key::PublicKey> {
// Node::object(
// serde_json::Value::new_object()
// .set_id(Some(&format!("{}#main-key", self.id))) // TODO is this some standard??
// .set_public_key_pem(&self.public_key)
// .set_owner(Some(&self.id))
// )
// }
// }
impl Model {
pub fn new(object: &impl Actor) -> Result<Self, super::FieldError> {
let ap_id = object.id().ok_or(super::FieldError("id"))?.to_string();
let (domain, preferred_username) = activitypub::split_id(&ap_id);
Ok(Model {
id: ap_id, preferred_username, domain,
actor_type: object.actor_type().ok_or(super::FieldError("type"))?,
name: object.name().map(|x| x.to_string()),
summary: object.summary().map(|x| x.to_string()),
icon: object.icon().id().map(|x| x.to_string()),
image: object.image().id().map(|x| x.to_string()),
inbox: object.inbox().id().map(|x| x.to_string()),
outbox: object.inbox().id().map(|x| x.to_string()),
shared_inbox: None, // TODO!!! parse endpoints
followers: object.followers().id().map(|x| x.to_string()),
following: object.following().id().map(|x| x.to_string()),
created: object.published().unwrap_or(chrono::Utc::now()),
updated: chrono::Utc::now(),
public_key: object.public_key().get().ok_or(super::FieldError("publicKey"))?.public_key_pem().to_string(),
private_key: None, // there's no way to transport privkey over AP json, must come from DB
})
}
}