mirror of
https://github.com/LemmyNet/lemmy.git
synced 2024-06-02 21:49:22 +00:00
1d38aad9d3
* a lot * merge * Fix stuff broken by merge * Get rid of repetitive `&mut *context.conn().await?` * Add blank lines under each line with `conn =` * Fix style mistakes (partial) * Revert "Fix style mistakes (partial)" This reverts commit48a033b87f
. * Revert "Add blank lines under each line with `conn =`" This reverts commit773a6d3beb
. * Revert "Get rid of repetitive `&mut *context.conn().await?`" This reverts commitd2c6263ea1
. * Use DbConn for CaptchaAnswer methods * DbConn trait * Remove more `&mut *` * Fix stuff * Re-run CI * try to make ci start * fix * fix * Fix api_common::utils * Fix apub::activities::block * Fix apub::api::resolve_object * Fix some things * Revert "Fix some things" This reverts commit2bf8574bc8
. * Revert "Fix apub::api::resolve_object" This reverts commit3e4059aabb
. * Revert "Fix apub::activities::block" This reverts commit3b02389abd
. * Revert "Fix api_common::utils" This reverts commit7dc73de613
. * Revert "Revert "Fix api_common::utils"" This reverts commitf740f115e5
. * Revert "Revert "Fix apub::activities::block"" This reverts commit2ee206af7c
. * Revert "Revert "Fix apub::api::resolve_object"" This reverts commit96ed8bf2e9
. * Fix fetch_local_site_data * Fix get_comment_parent_creator * Remove unused perma deleted text * Fix routes::feeds * Fix lib.rs * Update lib.rs * rerun ci * Attempt to create custom GetConn and RunQueryDsl traits * Start over * Add GetConn trait * aaaa * Revert "aaaa" This reverts commitacc9ca1aed
. * Revert "Revert "aaaa"" This reverts commit443a2a00a5
. * still aaaaaaaaaaaaa * Return to earlier thing Revert "Add GetConn trait" This reverts commitab4e94aea5
. * Try to use DbPool enum * Revert "Try to use DbPool enum" This reverts commite4d1712646
. * DbConn and DbPool enums (db_schema only fails to compile for tests) * fmt * Make functions take `&mut DbPool<'_>` and make db_schema tests compile * Add try_join_with_pool macro and run fix-clippy on more crates * Fix some errors * I did it * Remove function variants that take connection * rerun ci * rerun ci * rerun ci
187 lines
5.6 KiB
Rust
187 lines
5.6 KiB
Rust
use crate::PerformCrud;
|
|
use actix_web::web::Data;
|
|
use lemmy_api_common::{
|
|
build_response::build_post_response,
|
|
context::LemmyContext,
|
|
post::{CreatePost, PostResponse},
|
|
request::fetch_site_data,
|
|
utils::{
|
|
check_community_ban,
|
|
check_community_deleted_or_removed,
|
|
generate_local_apub_endpoint,
|
|
honeypot_check,
|
|
local_site_to_slur_regex,
|
|
local_user_view_from_jwt,
|
|
mark_post_as_read,
|
|
EndpointType,
|
|
},
|
|
};
|
|
use lemmy_db_schema::{
|
|
impls::actor_language::default_post_language,
|
|
source::{
|
|
actor_language::CommunityLanguage,
|
|
community::Community,
|
|
local_site::LocalSite,
|
|
post::{Post, PostInsertForm, PostLike, PostLikeForm, PostUpdateForm},
|
|
},
|
|
traits::{Crud, Likeable},
|
|
};
|
|
use lemmy_db_views_actor::structs::CommunityView;
|
|
use lemmy_utils::{
|
|
error::{LemmyError, LemmyErrorExt, LemmyErrorType},
|
|
spawn_try_task,
|
|
utils::{
|
|
slurs::{check_slurs, check_slurs_opt},
|
|
validation::{check_url_scheme, clean_url_params, is_valid_body_field, is_valid_post_title},
|
|
},
|
|
SYNCHRONOUS_FEDERATION,
|
|
};
|
|
use tracing::Instrument;
|
|
use url::Url;
|
|
use webmention::{Webmention, WebmentionError};
|
|
|
|
#[async_trait::async_trait(?Send)]
|
|
impl PerformCrud for CreatePost {
|
|
type Response = PostResponse;
|
|
|
|
#[tracing::instrument(skip(context))]
|
|
async fn perform(&self, context: &Data<LemmyContext>) -> Result<PostResponse, LemmyError> {
|
|
let data: &CreatePost = self;
|
|
let local_user_view = local_user_view_from_jwt(&data.auth, context).await?;
|
|
let local_site = LocalSite::read(&mut context.pool()).await?;
|
|
|
|
let slur_regex = local_site_to_slur_regex(&local_site);
|
|
check_slurs(&data.name, &slur_regex)?;
|
|
check_slurs_opt(&data.body, &slur_regex)?;
|
|
honeypot_check(&data.honeypot)?;
|
|
|
|
let data_url = data.url.as_ref();
|
|
let url = data_url.map(clean_url_params).map(Into::into); // TODO no good way to handle a "clear"
|
|
|
|
is_valid_post_title(&data.name)?;
|
|
is_valid_body_field(&data.body, true)?;
|
|
check_url_scheme(&data.url)?;
|
|
|
|
check_community_ban(
|
|
local_user_view.person.id,
|
|
data.community_id,
|
|
&mut context.pool(),
|
|
)
|
|
.await?;
|
|
check_community_deleted_or_removed(data.community_id, &mut context.pool()).await?;
|
|
|
|
let community_id = data.community_id;
|
|
let community = Community::read(&mut context.pool(), community_id).await?;
|
|
if community.posting_restricted_to_mods {
|
|
let community_id = data.community_id;
|
|
let is_mod = CommunityView::is_mod_or_admin(
|
|
&mut context.pool(),
|
|
local_user_view.local_user.person_id,
|
|
community_id,
|
|
)
|
|
.await?;
|
|
if !is_mod {
|
|
return Err(LemmyErrorType::OnlyModsCanPostInCommunity)?;
|
|
}
|
|
}
|
|
|
|
// Fetch post links and pictrs cached image
|
|
let (metadata_res, thumbnail_url) =
|
|
fetch_site_data(context.client(), context.settings(), data_url, true).await;
|
|
let (embed_title, embed_description, embed_video_url) = metadata_res
|
|
.map(|u| (u.title, u.description, u.embed_video_url))
|
|
.unwrap_or_default();
|
|
|
|
let language_id = match data.language_id {
|
|
Some(lid) => Some(lid),
|
|
None => {
|
|
default_post_language(
|
|
&mut context.pool(),
|
|
community_id,
|
|
local_user_view.local_user.id,
|
|
)
|
|
.await?
|
|
}
|
|
};
|
|
CommunityLanguage::is_allowed_community_language(
|
|
&mut context.pool(),
|
|
language_id,
|
|
community_id,
|
|
)
|
|
.await?;
|
|
|
|
let post_form = PostInsertForm::builder()
|
|
.name(data.name.trim().to_owned())
|
|
.url(url)
|
|
.body(data.body.clone())
|
|
.community_id(data.community_id)
|
|
.creator_id(local_user_view.person.id)
|
|
.nsfw(data.nsfw)
|
|
.embed_title(embed_title)
|
|
.embed_description(embed_description)
|
|
.embed_video_url(embed_video_url)
|
|
.language_id(language_id)
|
|
.thumbnail_url(thumbnail_url)
|
|
.build();
|
|
|
|
let inserted_post = Post::create(&mut context.pool(), &post_form)
|
|
.await
|
|
.with_lemmy_type(LemmyErrorType::CouldntCreatePost)?;
|
|
|
|
let inserted_post_id = inserted_post.id;
|
|
let protocol_and_hostname = context.settings().get_protocol_and_hostname();
|
|
let apub_id = generate_local_apub_endpoint(
|
|
EndpointType::Post,
|
|
&inserted_post_id.to_string(),
|
|
&protocol_and_hostname,
|
|
)?;
|
|
let updated_post = Post::update(
|
|
&mut context.pool(),
|
|
inserted_post_id,
|
|
&PostUpdateForm::builder().ap_id(Some(apub_id)).build(),
|
|
)
|
|
.await
|
|
.with_lemmy_type(LemmyErrorType::CouldntCreatePost)?;
|
|
|
|
// They like their own post by default
|
|
let person_id = local_user_view.person.id;
|
|
let post_id = inserted_post.id;
|
|
let like_form = PostLikeForm {
|
|
post_id,
|
|
person_id,
|
|
score: 1,
|
|
};
|
|
|
|
PostLike::like(&mut context.pool(), &like_form)
|
|
.await
|
|
.with_lemmy_type(LemmyErrorType::CouldntLikePost)?;
|
|
|
|
// Mark the post as read
|
|
mark_post_as_read(person_id, post_id, &mut context.pool()).await?;
|
|
|
|
if let Some(url) = updated_post.url.clone() {
|
|
let task = async move {
|
|
let mut webmention =
|
|
Webmention::new::<Url>(updated_post.ap_id.clone().into(), url.clone().into())?;
|
|
webmention.set_checked(true);
|
|
match webmention
|
|
.send()
|
|
.instrument(tracing::info_span!("Sending webmention"))
|
|
.await
|
|
{
|
|
Err(WebmentionError::NoEndpointDiscovered(_)) => Ok(()),
|
|
Ok(_) => Ok(()),
|
|
Err(e) => Err(e).with_lemmy_type(LemmyErrorType::CouldntSendWebmention),
|
|
}
|
|
};
|
|
if *SYNCHRONOUS_FEDERATION {
|
|
task.await?;
|
|
} else {
|
|
spawn_try_task(task);
|
|
}
|
|
};
|
|
|
|
build_post_response(context, community_id, person_id, post_id).await
|
|
}
|
|
}
|