2018-09-04 10:37:58 +00:00
use activitypub ::{ Actor , activity ::{ Accept , Follow as FollowAct , Undo } , actor ::Person } ;
2018-05-01 13:06:31 +00:00
use diesel ::{ self , PgConnection , ExpressionMethods , QueryDsl , RunQueryDsl } ;
2018-09-04 10:37:58 +00:00
use plume_common ::activity_pub ::{ broadcast , Id , IntoId , inbox ::{ FromActivity , Notify , WithInbox , Deletable } , sign ::Signer } ;
2018-06-23 16:36:11 +00:00
use blogs ::Blog ;
use notifications ::* ;
use users ::User ;
2018-05-01 13:06:31 +00:00
use schema ::follows ;
2018-05-01 13:23:23 +00:00
#[ derive(Queryable, Identifiable, Associations) ]
#[ belongs_to(User, foreign_key = " following_id " ) ]
2018-05-01 13:06:31 +00:00
pub struct Follow {
pub id : i32 ,
pub follower_id : i32 ,
2018-09-04 10:37:58 +00:00
pub following_id : i32 ,
pub ap_url : String ,
2018-05-01 13:06:31 +00:00
}
#[ derive(Insertable) ]
#[ table_name = " follows " ]
pub struct NewFollow {
pub follower_id : i32 ,
2018-09-04 10:37:58 +00:00
pub following_id : i32 ,
pub ap_url : String ,
2018-05-01 13:06:31 +00:00
}
impl Follow {
2018-06-18 13:57:38 +00:00
insert! ( follows , NewFollow ) ;
2018-06-18 13:44:23 +00:00
get! ( follows ) ;
2018-09-04 10:37:58 +00:00
find_by! ( follows , find_by_ap_url , ap_url as String ) ;
pub fn find ( conn : & PgConnection , from : i32 , to : i32 ) -> Option < Follow > {
follows ::table . filter ( follows ::follower_id . eq ( from ) )
. filter ( follows ::following_id . eq ( to ) )
. get_result ( conn )
. ok ( )
}
pub fn into_activity ( & self , conn : & PgConnection ) -> FollowAct {
let user = User ::get ( conn , self . follower_id ) . unwrap ( ) ;
let target = User ::get ( conn , self . following_id ) . unwrap ( ) ;
let mut act = FollowAct ::default ( ) ;
act . follow_props . set_actor_link ::< Id > ( user . clone ( ) . into_id ( ) ) . expect ( " Follow::into_activity: actor error " ) ;
act . follow_props . set_object_object ( user . into_activity ( & * conn ) ) . unwrap ( ) ;
act . object_props . set_id_string ( self . ap_url . clone ( ) ) . unwrap ( ) ;
act . object_props . set_to_link ( target . clone ( ) . into_id ( ) ) . expect ( " New Follow error while setting 'to' " ) ;
act . object_props . set_cc_link_vec ::< Id > ( vec! [ ] ) . expect ( " New Follow error while setting 'cc' " ) ;
act
}
2018-06-12 19:10:08 +00:00
2018-06-22 15:17:53 +00:00
/// from -> The one sending the follow request
/// target -> The target of the request, responding with Accept
2018-06-23 11:50:14 +00:00
pub fn accept_follow < A : Signer + IntoId + Clone , B : Clone + WithInbox + Actor + IntoId > (
2018-06-12 19:10:08 +00:00
conn : & PgConnection ,
2018-06-22 15:17:53 +00:00
from : & B ,
target : & A ,
2018-06-12 19:10:08 +00:00
follow : FollowAct ,
from_id : i32 ,
target_id : i32
) -> Follow {
2018-09-04 10:37:58 +00:00
let from_url : String = from . clone ( ) . into_id ( ) . into ( ) ;
let target_url : String = target . clone ( ) . into_id ( ) . into ( ) ;
2018-06-12 19:10:08 +00:00
let res = Follow ::insert ( conn , NewFollow {
follower_id : from_id ,
2018-09-04 10:37:58 +00:00
following_id : target_id ,
ap_url : format ! ( " {}/follow/{} " , from_url , target_url ) ,
2018-06-12 19:10:08 +00:00
} ) ;
let mut accept = Accept ::default ( ) ;
2018-06-23 11:50:14 +00:00
let accept_id = format! ( " {} #accept " , follow . object_props . id_string ( ) . unwrap_or ( String ::new ( ) ) ) ;
accept . object_props . set_id_string ( accept_id ) . expect ( " accept_follow: id error " ) ;
accept . object_props . set_to_link ( from . clone ( ) . into_id ( ) ) . expect ( " accept_follow: to error " ) ;
accept . object_props . set_cc_link_vec ::< Id > ( vec! [ ] ) . expect ( " accept_follow: cc error " ) ;
2018-06-22 15:17:53 +00:00
accept . accept_props . set_actor_link ::< Id > ( target . clone ( ) . into_id ( ) ) . unwrap ( ) ;
2018-06-12 19:10:08 +00:00
accept . accept_props . set_object_object ( follow ) . unwrap ( ) ;
2018-06-22 15:17:53 +00:00
broadcast ( & * target , accept , vec! [ from . clone ( ) ] ) ;
2018-06-12 19:10:08 +00:00
res
}
}
2018-06-23 16:36:11 +00:00
impl FromActivity < FollowAct , PgConnection > for Follow {
2018-06-12 19:10:08 +00:00
fn from_activity ( conn : & PgConnection , follow : FollowAct , _actor : Id ) -> Follow {
2018-07-08 18:01:19 +00:00
let from_id = follow . follow_props . actor_link ::< Id > ( ) . map ( | l | l . into ( ) )
. unwrap_or_else ( | _ | follow . follow_props . actor_object ::< Person > ( ) . expect ( " No actor object (nor ID) on Follow " ) . object_props . id_string ( ) . expect ( " No ID on actor on Follow " ) ) ;
let from = User ::from_url ( conn , from_id ) . unwrap ( ) ;
2018-06-12 19:10:08 +00:00
match User ::from_url ( conn , follow . follow_props . object . as_str ( ) . unwrap ( ) . to_string ( ) ) {
2018-06-22 15:17:53 +00:00
Some ( user ) = > Follow ::accept_follow ( conn , & from , & user , follow , from . id , user . id ) ,
2018-06-12 19:10:08 +00:00
None = > {
let blog = Blog ::from_url ( conn , follow . follow_props . object . as_str ( ) . unwrap ( ) . to_string ( ) ) . unwrap ( ) ;
Follow ::accept_follow ( conn , & from , & blog , follow , from . id , blog . id )
}
}
}
2018-05-01 13:06:31 +00:00
}
2018-06-17 19:37:10 +00:00
2018-06-23 16:36:11 +00:00
impl Notify < PgConnection > for Follow {
2018-06-20 21:51:47 +00:00
fn notify ( & self , conn : & PgConnection ) {
2018-06-17 19:37:10 +00:00
Notification ::insert ( conn , NewNotification {
2018-07-26 13:46:10 +00:00
kind : notification_kind ::FOLLOW . to_string ( ) ,
object_id : self . id ,
2018-06-20 21:51:47 +00:00
user_id : self . following_id
2018-06-17 19:37:10 +00:00
} ) ;
}
}
2018-09-04 10:37:58 +00:00
impl Deletable < PgConnection , Undo > for Follow {
fn delete ( & self , conn : & PgConnection ) -> Undo {
diesel ::delete ( self ) . execute ( conn ) . expect ( " Coudn't delete follow " ) ;
2018-09-07 22:29:50 +00:00
// delete associated notification if any
if let Some ( notif ) = Notification ::find ( conn , notification_kind ::FOLLOW , self . id ) {
diesel ::delete ( & notif ) . execute ( conn ) . expect ( " Couldn't delete follow notification " ) ;
}
2018-09-04 10:37:58 +00:00
let mut undo = Undo ::default ( ) ;
undo . undo_props . set_actor_link ( User ::get ( conn , self . follower_id ) . unwrap ( ) . into_id ( ) ) . expect ( " Follow::delete: actor error " ) ;
undo . object_props . set_id_string ( format! ( " {} /undo " , self . ap_url ) ) . expect ( " Follow::delete: id error " ) ;
undo . undo_props . set_object_object ( self . into_activity ( conn ) ) . expect ( " Follow::delete: object error " ) ;
undo
}
fn delete_id ( id : String , conn : & PgConnection ) {
if let Some ( follow ) = Follow ::find_by_ap_url ( conn , id ) {
follow . delete ( conn ) ;
}
}
}