mirror of
https://github.com/versia-pub/activitypub.git
synced 2025-12-06 06:38:20 +01:00
247 lines
6.9 KiB
Rust
247 lines
6.9 KiB
Rust
use activitypub_federation::{
|
|
fetch::{object_id::ObjectId, webfinger::webfinger_resolve_actor},
|
|
protocol::{context::WithContext, public_key::PublicKey},
|
|
traits::Object,
|
|
FEDERATION_CONTENT_TYPE,
|
|
};
|
|
use activitystreams_kinds::{activity::CreateType, object};
|
|
use actix_web::{get, post, web, HttpResponse};
|
|
use sea_orm::{query, ColumnTrait, EntityTrait, QueryFilter};
|
|
use url::Url;
|
|
|
|
use crate::{
|
|
database::State,
|
|
entities::{
|
|
post::{self, Entity},
|
|
prelude, user,
|
|
},
|
|
error,
|
|
lysand::{
|
|
conversion::{lysand_post_from_db, lysand_user_from_db},
|
|
inbox::inbox_entry,
|
|
},
|
|
objects::{self, person::Person},
|
|
utils::{base_url_decode, generate_create_id, generate_user_id},
|
|
Response, API_DOMAIN, DB, FEDERATION_CONFIG,
|
|
};
|
|
|
|
#[derive(serde::Deserialize)]
|
|
struct LysandQuery {
|
|
// Post url
|
|
url: Option<Url>,
|
|
// User handle
|
|
user: Option<String>,
|
|
// User URL
|
|
user_url: Option<Url>,
|
|
}
|
|
|
|
#[get("/apbridge/lysand/query")]
|
|
async fn query_post(
|
|
query: web::Query<LysandQuery>,
|
|
state: web::Data<State>,
|
|
) -> actix_web::Result<HttpResponse, error::Error> {
|
|
if query.url.is_none() && query.user.is_none() && query.user_url.is_none() {
|
|
return Ok(
|
|
HttpResponse::BadRequest().body("Bad Request. Error code: mrrrmrrrmrrawwawwawwa")
|
|
);
|
|
}
|
|
|
|
let db = DB.get().unwrap();
|
|
let data = FEDERATION_CONFIG.get().unwrap();
|
|
|
|
if let Some(user) = query.user.clone() {
|
|
let target =
|
|
webfinger_resolve_actor::<State, user::Model>(user.as_str(), &data.to_request_data())
|
|
.await?;
|
|
let lysand_user = lysand_user_from_db(target).await?;
|
|
|
|
return Ok(HttpResponse::Ok()
|
|
.content_type("application/json")
|
|
.json(lysand_user));
|
|
}
|
|
|
|
if let Some(user) = query.user_url.clone() {
|
|
let lysand_user = lysand_url_to_user(user).await?;
|
|
|
|
return Ok(HttpResponse::Ok()
|
|
.content_type("application/json")
|
|
.json(lysand_user));
|
|
}
|
|
|
|
let opt_model = prelude::Post::find()
|
|
.filter(post::Column::Url.eq(query.url.clone().unwrap().as_str()))
|
|
.one(db)
|
|
.await?;
|
|
let target;
|
|
if let Some(model) = opt_model {
|
|
target = model;
|
|
} else {
|
|
target = ObjectId::<post::Model>::from(Url::parse(query.url.clone().unwrap().as_str())?)
|
|
.dereference(&data.to_request_data())
|
|
.await?;
|
|
}
|
|
|
|
Ok(HttpResponse::Ok()
|
|
.content_type("application/json")
|
|
.json(lysand_post_from_db(target).await?))
|
|
}
|
|
|
|
#[post("/apbridge/lysand/inbox")]
|
|
async fn lysand_inbox(
|
|
body: web::Bytes,
|
|
state: web::Data<State>,
|
|
) -> actix_web::Result<HttpResponse, error::Error> {
|
|
let string = String::from_utf8(body.to_vec())?;
|
|
inbox_entry(&string).await?;
|
|
Ok(HttpResponse::Created().finish())
|
|
}
|
|
|
|
#[get("/apbridge/object/{post}")]
|
|
async fn fetch_post(
|
|
path: web::Path<String>,
|
|
state: web::Data<State>,
|
|
) -> actix_web::Result<HttpResponse, error::Error> {
|
|
let db = DB.get().unwrap();
|
|
|
|
let post = prelude::Post::find()
|
|
.filter(post::Column::Id.eq(path.as_str()))
|
|
.one(db)
|
|
.await?;
|
|
|
|
let post = match post {
|
|
Some(post) => post,
|
|
None => return Ok(HttpResponse::NotFound().finish()),
|
|
};
|
|
|
|
Ok(HttpResponse::Ok()
|
|
.content_type(FEDERATION_CONTENT_TYPE)
|
|
.json(crate::objects::post::Note::from_db(&post)))
|
|
}
|
|
|
|
#[get("/apbridge/user/{user}")]
|
|
async fn fetch_user(
|
|
path: web::Path<String>,
|
|
state: web::Data<State>,
|
|
) -> actix_web::Result<HttpResponse, error::Error> {
|
|
let db = DB.get().unwrap();
|
|
|
|
let user = prelude::User::find()
|
|
.filter(user::Column::Id.eq(path.as_str()))
|
|
.one(db)
|
|
.await?;
|
|
|
|
let user = match user {
|
|
Some(user) => user,
|
|
None => return Ok(HttpResponse::NotFound().finish()),
|
|
};
|
|
|
|
let deserialized_user: Person = serde_json::from_str(user.ap_json.as_ref().unwrap().as_str())?;
|
|
|
|
Ok(HttpResponse::Ok()
|
|
.content_type(FEDERATION_CONTENT_TYPE)
|
|
.json(WithContext::new_default(deserialized_user)))
|
|
}
|
|
|
|
#[get("/apbridge/lysand/object/{post}")]
|
|
async fn fetch_lysand_post(
|
|
path: web::Path<String>,
|
|
state: web::Data<State>,
|
|
) -> actix_web::Result<HttpResponse, error::Error> {
|
|
let db = DB.get().unwrap();
|
|
|
|
let post = prelude::Post::find()
|
|
.filter(post::Column::Id.eq(path.as_str()))
|
|
.one(db)
|
|
.await?;
|
|
|
|
let post = match post {
|
|
Some(post) => post,
|
|
None => return Ok(HttpResponse::NotFound().finish()),
|
|
};
|
|
|
|
Ok(HttpResponse::Ok()
|
|
.content_type("application/json")
|
|
.json(lysand_post_from_db(post).await?))
|
|
}
|
|
|
|
#[get("/apbridge/create/{id}/{base64url}")]
|
|
async fn create_activity(
|
|
path: web::Path<(String, String)>,
|
|
state: web::Data<State>,
|
|
) -> actix_web::Result<HttpResponse, error::Error> {
|
|
let db = DB.get().unwrap();
|
|
|
|
let url = base_url_decode(path.1.as_str());
|
|
|
|
let post = prelude::Post::find()
|
|
.filter(post::Column::Id.eq(path.0.as_str()))
|
|
.one(db)
|
|
.await?;
|
|
|
|
let post = match post {
|
|
Some(post) => post,
|
|
None => return Ok(HttpResponse::NotFound().finish()),
|
|
};
|
|
|
|
let ap_post = crate::objects::post::Note::from_db(&post);
|
|
|
|
let data = FEDERATION_CONFIG.get().unwrap();
|
|
|
|
let create = crate::activities::create_post::CreatePost {
|
|
actor: ap_post.attributed_to.clone(),
|
|
to: ap_post.to.clone(),
|
|
object: ap_post,
|
|
kind: CreateType::Create,
|
|
id: generate_create_id(&data.to_request_data().domain(), &path.0, &path.1)?,
|
|
};
|
|
let create_with_context = WithContext::new_default(create);
|
|
|
|
Ok(HttpResponse::Ok()
|
|
.content_type(FEDERATION_CONTENT_TYPE)
|
|
.json(create_with_context))
|
|
}
|
|
|
|
pub async fn lysand_url_to_user(url: Url) -> anyhow::Result<super::objects::User> {
|
|
let db = DB.get().unwrap();
|
|
let data = FEDERATION_CONFIG.get().unwrap();
|
|
|
|
let opt_model = prelude::User::find()
|
|
.filter(user::Column::Url.eq(url.as_str()))
|
|
.one(db)
|
|
.await?;
|
|
let target;
|
|
if let Some(model) = opt_model {
|
|
target = model;
|
|
} else {
|
|
target = ObjectId::<user::Model>::from(url)
|
|
.dereference(&data.to_request_data())
|
|
.await
|
|
.unwrap();
|
|
}
|
|
|
|
Ok(lysand_user_from_db(target).await?)
|
|
}
|
|
|
|
pub async fn lysand_url_to_user_and_model(
|
|
url: Url,
|
|
) -> anyhow::Result<(super::objects::User, user::Model)> {
|
|
let db = DB.get().unwrap();
|
|
let data = FEDERATION_CONFIG.get().unwrap();
|
|
|
|
let opt_model = prelude::User::find()
|
|
.filter(user::Column::Url.eq(url.as_str()))
|
|
.one(db)
|
|
.await?;
|
|
let target;
|
|
if let Some(model) = opt_model {
|
|
target = model;
|
|
} else {
|
|
target = ObjectId::<user::Model>::from(url)
|
|
.dereference(&data.to_request_data())
|
|
.await
|
|
.unwrap();
|
|
}
|
|
|
|
Ok((lysand_user_from_db(target.clone()).await?, target))
|
|
}
|