2017-03-21 12:53:20 -04:00
defmodule Pleroma.Web.TwitterAPI.TwitterAPITest do
use Pleroma.DataCase
alias Pleroma.Builders . { UserBuilder , ActivityBuilder }
alias Pleroma.Web.TwitterAPI.TwitterAPI
2017-05-17 12:00:20 -04:00
alias Pleroma.Web.TwitterAPI.Utils
2017-03-30 11:07:03 -04:00
alias Pleroma . { Activity , User , Object , Repo }
2017-04-16 04:25:27 -04:00
alias Pleroma.Web.TwitterAPI.Representers . { ActivityRepresenter , UserRepresenter }
2017-04-14 12:15:15 -04:00
alias Pleroma.Web.ActivityPub.ActivityPub
2017-03-21 12:53:20 -04:00
2017-04-13 10:19:07 -04:00
import Pleroma.Factory
2017-03-21 12:53:20 -04:00
test " create a status " do
2017-04-03 12:28:19 -04:00
user = UserBuilder . build ( %{ ap_id : " 142344 " } )
_mentioned_user = UserBuilder . insert ( %{ nickname : " shp " , ap_id : " shp " } )
2017-03-30 11:07:03 -04:00
object_data = %{
" type " = > " Image " ,
" url " = > [
%{
" type " = > " Link " ,
" mediaType " = > " image/jpg " ,
" href " = > " http://example.org/image.jpg "
}
] ,
" uuid " = > 1
}
object = Repo . insert! ( % Object { data : object_data } )
2017-03-21 12:53:20 -04:00
input = %{
2017-05-18 09:16:49 -04:00
" status " = > " Hello again, @shp.<script></script> \n This is on another line. # 2hu # epic # phantasmagoric " ,
2017-03-30 11:07:03 -04:00
" media_ids " = > [ object . id ]
2017-03-21 12:53:20 -04:00
}
{ :ok , activity = % Activity { } } = TwitterAPI . create_status ( user , input )
2017-05-18 09:16:49 -04:00
assert get_in ( activity . data , [ " object " , " content " ] ) == " Hello again, <a href='shp'>@shp</a>.<br>This is on another line. # 2hu # epic # phantasmagoric<br><a href='http://example.org/image.jpg' class='attachment'>image.jpg</a> "
2017-03-21 13:17:35 -04:00
assert get_in ( activity . data , [ " object " , " type " ] ) == " Note "
2017-04-13 08:56:19 -04:00
assert get_in ( activity . data , [ " object " , " actor " ] ) == user . ap_id
2017-04-03 12:28:19 -04:00
assert get_in ( activity . data , [ " actor " ] ) == user . ap_id
2017-03-21 13:17:35 -04:00
assert Enum . member? ( get_in ( activity . data , [ " to " ] ) , User . ap_followers ( user ) )
assert Enum . member? ( get_in ( activity . data , [ " to " ] ) , " https://www.w3.org/ns/activitystreams # Public " )
2017-04-03 12:28:19 -04:00
assert Enum . member? ( get_in ( activity . data , [ " to " ] ) , " shp " )
2017-05-02 04:43:35 -04:00
assert activity . local == true
2017-03-23 18:34:10 -04:00
2017-05-18 09:16:49 -04:00
# hashtags
assert activity . data [ " object " ] [ " tag " ] == [ " 2hu " , " epic " , " phantasmagoric " ]
2017-04-30 08:02:04 -04:00
# Add a context
2017-03-23 18:34:10 -04:00
assert is_binary ( get_in ( activity . data , [ " context " ] ) )
assert is_binary ( get_in ( activity . data , [ " object " , " context " ] ) )
2017-03-30 11:07:03 -04:00
2017-03-30 12:07:38 -04:00
assert is_list ( activity . data [ " object " ] [ " attachment " ] )
2017-04-14 12:59:11 -04:00
assert activity . data [ " object " ] == Object . get_by_ap_id ( activity . data [ " object " ] [ " id " ] ) . data
2017-03-23 18:34:10 -04:00
end
test " create a status that is a reply " do
2017-04-03 12:28:19 -04:00
user = UserBuilder . build ( %{ ap_id : " some_cool_id " } )
2017-03-23 18:34:10 -04:00
input = %{
" status " = > " Hello again. "
}
{ :ok , activity = % Activity { } } = TwitterAPI . create_status ( user , input )
input = %{
" status " = > " Here's your (you). " ,
" in_reply_to_status_id " = > activity . id
}
{ :ok , reply = % Activity { } } = TwitterAPI . create_status ( user , input )
assert get_in ( reply . data , [ " context " ] ) == get_in ( activity . data , [ " context " ] )
assert get_in ( reply . data , [ " object " , " context " ] ) == get_in ( activity . data , [ " object " , " context " ] )
assert get_in ( reply . data , [ " object " , " inReplyTo " ] ) == get_in ( activity . data , [ " object " , " id " ] )
assert get_in ( reply . data , [ " object " , " inReplyToStatusId " ] ) == activity . id
2017-04-03 12:28:19 -04:00
assert Enum . member? ( get_in ( reply . data , [ " to " ] ) , " some_cool_id " )
2017-03-21 12:53:20 -04:00
end
2017-05-02 08:12:43 -04:00
test " fetch public statuses, excluding remote ones. " do
2017-03-21 12:53:20 -04:00
%{ public : activity , user : user } = ActivityBuilder . public_and_non_public
2017-05-02 08:12:43 -04:00
insert ( :note_activity , %{ local : false } )
2017-04-16 09:28:28 -04:00
follower = insert ( :user , following : [ User . ap_followers ( user ) ] )
2017-03-23 10:51:34 -04:00
statuses = TwitterAPI . fetch_public_statuses ( follower )
2017-03-21 12:53:20 -04:00
assert length ( statuses ) == 1
2017-03-23 10:51:34 -04:00
assert Enum . at ( statuses , 0 ) == ActivityRepresenter . to_map ( activity , %{ user : user , for : follower } )
2017-03-21 12:53:20 -04:00
end
2017-03-22 11:51:20 -04:00
2017-05-02 08:12:43 -04:00
test " fetch whole known network statuses " do
%{ public : activity , user : user } = ActivityBuilder . public_and_non_public
insert ( :note_activity , %{ local : false } )
follower = insert ( :user , following : [ User . ap_followers ( user ) ] )
statuses = TwitterAPI . fetch_public_and_external_statuses ( follower )
assert length ( statuses ) == 2
assert Enum . at ( statuses , 0 ) == ActivityRepresenter . to_map ( activity , %{ user : user , for : follower } )
end
2017-03-22 11:51:20 -04:00
test " fetch friends' statuses " do
2017-04-16 09:28:28 -04:00
user = insert ( :user , %{ following : [ " someguy/followers " ] } )
2017-03-22 11:51:20 -04:00
{ :ok , activity } = ActivityBuilder . insert ( %{ " to " = > [ " someguy/followers " ] } )
2017-04-16 09:28:28 -04:00
{ :ok , direct_activity } = ActivityBuilder . insert ( %{ " to " = > [ user . ap_id ] } )
2017-03-22 11:51:20 -04:00
statuses = TwitterAPI . fetch_friend_statuses ( user )
activity_user = Repo . get_by ( User , ap_id : activity . data [ " actor " ] )
2017-04-16 09:28:28 -04:00
direct_activity_user = Repo . get_by ( User , ap_id : direct_activity . data [ " actor " ] )
2017-03-22 11:51:20 -04:00
2017-04-12 10:45:23 -04:00
assert length ( statuses ) == 2
2017-03-22 11:51:20 -04:00
assert Enum . at ( statuses , 0 ) == ActivityRepresenter . to_map ( activity , %{ user : activity_user } )
2017-04-16 09:28:28 -04:00
assert Enum . at ( statuses , 1 ) == ActivityRepresenter . to_map ( direct_activity , %{ user : direct_activity_user , mentioned : [ user ] } )
2017-03-22 11:51:20 -04:00
end
2017-03-22 13:36:08 -04:00
2017-04-20 06:53:53 -04:00
test " fetch user's mentions " do
user = insert ( :user )
{ :ok , activity } = ActivityBuilder . insert ( %{ " to " = > [ user . ap_id ] } )
activity_user = Repo . get_by ( User , ap_id : activity . data [ " actor " ] )
statuses = TwitterAPI . fetch_mentions ( user )
assert length ( statuses ) == 1
assert Enum . at ( statuses , 0 ) == ActivityRepresenter . to_map ( activity , %{ user : activity_user , mentioned : [ user ] } )
2017-03-22 11:51:20 -04:00
end
2017-03-22 13:36:08 -04:00
2017-04-16 09:44:30 -04:00
test " get a user by params " do
user1_result = { :ok , user1 } = UserBuilder . insert ( %{ ap_id : " some id " , email : " test@pleroma " } )
{ :ok , user2 } = UserBuilder . insert ( %{ ap_id : " some other id " , nickname : " testname2 " , email : " test2@pleroma " } )
2017-04-16 10:05:48 -04:00
assert { :error , " You need to specify screen_name or user_id " } == TwitterAPI . get_user ( nil , nil )
2017-04-16 09:44:30 -04:00
assert user1_result == TwitterAPI . get_user ( nil , %{ " user_id " = > user1 . id } )
assert user1_result == TwitterAPI . get_user ( nil , %{ " screen_name " = > user1 . nickname } )
assert user1_result == TwitterAPI . get_user ( user1 , nil )
assert user1_result == TwitterAPI . get_user ( user2 , %{ " user_id " = > user1 . id } )
assert user1_result == TwitterAPI . get_user ( user2 , %{ " screen_name " = > user1 . nickname } )
assert { :error , " No user with such screen_name " } == TwitterAPI . get_user ( nil , %{ " screen_name " = > " Satan " } )
assert { :error , " No user with such user_id " } == TwitterAPI . get_user ( nil , %{ " user_id " = > 666 } )
end
2017-04-14 09:09:13 -04:00
test " fetch user's statuses " do
2017-04-16 09:44:30 -04:00
{ :ok , user1 } = UserBuilder . insert ( %{ ap_id : " some id " , email : " test@pleroma " } )
{ :ok , user2 } = UserBuilder . insert ( %{ ap_id : " some other id " , nickname : " testname2 " , email : " test2@pleroma " } )
2017-04-14 09:09:13 -04:00
{ :ok , status1 } = ActivityBuilder . insert ( %{ " id " = > 1 } , %{ user : user1 } )
{ :ok , status2 } = ActivityBuilder . insert ( %{ " id " = > 2 } , %{ user : user2 } )
2017-04-16 09:44:30 -04:00
user1_statuses = TwitterAPI . fetch_user_statuses ( user1 , %{ " actor_id " = > user1 . ap_id } )
2017-04-14 09:09:13 -04:00
assert length ( user1_statuses ) == 1
assert Enum . at ( user1_statuses , 0 ) == ActivityRepresenter . to_map ( status1 , %{ user : user1 } )
2017-04-16 09:44:30 -04:00
user2_statuses = TwitterAPI . fetch_user_statuses ( user1 , %{ " actor_id " = > user2 . ap_id } )
2017-04-14 09:09:13 -04:00
assert length ( user2_statuses ) == 1
assert Enum . at ( user2_statuses , 0 ) == ActivityRepresenter . to_map ( status2 , %{ user : user2 } )
end
2017-03-23 20:16:28 -04:00
test " fetch a single status " do
{ :ok , activity } = ActivityBuilder . insert ( )
{ :ok , user } = UserBuilder . insert ( )
actor = Repo . get_by! ( User , ap_id : activity . data [ " actor " ] )
status = TwitterAPI . fetch_status ( user , activity . id )
assert status == ActivityRepresenter . to_map ( activity , %{ for : user , user : actor } )
end
2017-04-10 09:38:21 -04:00
test " Follow another user using user_id " do
2017-04-16 09:28:28 -04:00
user = insert ( :user )
2017-04-16 10:18:34 -04:00
followed = insert ( :user )
2017-03-22 13:36:08 -04:00
2017-04-20 03:39:18 -04:00
{ :ok , user , followed , _activity } = TwitterAPI . follow ( user , %{ " user_id " = > followed . id } )
assert user . following == [ User . ap_followers ( followed ) ]
2017-03-22 13:36:08 -04:00
2017-04-20 03:39:18 -04:00
{ :error , msg } = TwitterAPI . follow ( user , %{ " user_id " = > followed . id } )
assert msg == " Could not follow user: #{ followed . nickname } is already on your list. "
2017-04-10 09:38:21 -04:00
end
test " Follow another user using screen_name " do
2017-04-16 10:12:28 -04:00
user = insert ( :user )
2017-04-20 03:39:18 -04:00
followed = insert ( :user )
2017-03-22 13:36:08 -04:00
2017-04-20 03:39:18 -04:00
{ :ok , user , followed , _activity } = TwitterAPI . follow ( user , %{ " screen_name " = > followed . nickname } )
2017-04-12 10:34:36 -04:00
assert user . following == [ User . ap_followers ( followed ) ]
2017-04-20 03:39:18 -04:00
{ :error , msg } = TwitterAPI . follow ( user , %{ " screen_name " = > followed . nickname } )
2017-04-12 10:34:36 -04:00
assert msg == " Could not follow user: #{ followed . nickname } is already on your list. "
2017-03-22 13:36:08 -04:00
end
2017-03-23 08:13:09 -04:00
2017-04-20 03:46:27 -04:00
test " Unfollow another user using user_id " do
unfollowed = insert ( :user )
user = insert ( :user , %{ following : [ User . ap_followers ( unfollowed ) ] } )
2017-05-07 13:28:23 -04:00
ActivityPub . follow ( user , unfollowed )
2017-04-10 09:45:47 -04:00
2017-04-20 03:46:27 -04:00
{ :ok , user , unfollowed } = TwitterAPI . unfollow ( user , %{ " user_id " = > unfollowed . id } )
2017-04-10 09:45:47 -04:00
assert user . following == [ ]
2017-03-23 08:13:09 -04:00
2017-04-20 03:46:27 -04:00
{ :error , msg } = TwitterAPI . unfollow ( user , %{ " user_id " = > unfollowed . id } )
assert msg == " Not subscribed! "
2017-04-10 09:45:47 -04:00
end
2017-03-23 08:13:09 -04:00
2017-04-10 09:45:47 -04:00
test " Unfollow another user using screen_name " do
2017-04-20 03:46:27 -04:00
unfollowed = insert ( :user )
user = insert ( :user , %{ following : [ User . ap_followers ( unfollowed ) ] } )
2017-03-23 08:13:09 -04:00
2017-05-07 13:28:23 -04:00
ActivityPub . follow ( user , unfollowed )
2017-04-20 03:46:27 -04:00
{ :ok , user , unfollowed } = TwitterAPI . unfollow ( user , %{ " screen_name " = > unfollowed . nickname } )
2017-03-23 08:13:09 -04:00
assert user . following == [ ]
2017-04-20 03:46:27 -04:00
{ :error , msg } = TwitterAPI . unfollow ( user , %{ " screen_name " = > unfollowed . nickname } )
2017-04-12 10:34:36 -04:00
assert msg == " Not subscribed! "
2017-03-23 08:13:09 -04:00
end
2017-03-28 11:22:44 -04:00
test " fetch statuses in a context using the conversation id " do
{ :ok , user } = UserBuilder . insert ( )
2017-04-30 07:53:26 -04:00
{ :ok , activity } = ActivityBuilder . insert ( %{ " context " = > " 2hu " } )
{ :ok , activity_two } = ActivityBuilder . insert ( %{ " context " = > " 2hu " } )
2017-03-28 11:22:44 -04:00
{ :ok , _activity_three } = ActivityBuilder . insert ( %{ " context " = > " 3hu " } )
2017-04-30 07:53:26 -04:00
{ :ok , object } = Object . context_mapping ( " 2hu " ) |> Repo . insert
statuses = TwitterAPI . fetch_conversation ( user , object . id )
2017-03-28 11:22:44 -04:00
assert length ( statuses ) == 2
assert Enum . at ( statuses , 0 ) [ " id " ] == activity . id
assert Enum . at ( statuses , 1 ) [ " id " ] == activity_two . id
end
2017-03-28 20:05:51 -04:00
test " upload a file " do
file = % Plug.Upload { content_type : " image/jpg " , path : Path . absname ( " test/fixtures/image.jpg " ) , filename : " an_image.jpg " }
response = TwitterAPI . upload ( file )
assert is_binary ( response )
end
2017-04-03 12:28:19 -04:00
test " it can parse mentions and return the relevant users " do
2017-05-15 12:25:21 -04:00
text = " @gsimg According to @archaeme, that is @daggsy. Also hello @archaeme@archae.me "
2017-04-03 12:28:19 -04:00
2017-04-16 09:28:28 -04:00
gsimg = insert ( :user , %{ nickname : " gsimg " } )
archaeme = insert ( :user , %{ nickname : " archaeme " } )
2017-05-15 12:25:21 -04:00
archaeme_remote = insert ( :user , %{ nickname : " archaeme@archae.me " } )
2017-04-03 12:28:19 -04:00
expected_result = [
{ " @gsimg " , gsimg } ,
2017-05-15 12:25:21 -04:00
{ " @archaeme " , archaeme } ,
{ " @archaeme@archae.me " , archaeme_remote } ,
2017-04-03 12:28:19 -04:00
]
assert TwitterAPI . parse_mentions ( text ) == expected_result
end
test " it adds user links to an existing text " do
2017-05-15 12:25:21 -04:00
text = " @gsimg According to @archaeme, that is @daggsy. Also hello @archaeme@archae.me "
2017-04-03 12:28:19 -04:00
2017-04-16 09:28:28 -04:00
gsimg = insert ( :user , %{ nickname : " gsimg " } )
archaeme = insert ( :user , %{ nickname : " archaeme " } )
2017-05-15 12:25:21 -04:00
archaeme_remote = insert ( :user , %{ nickname : " archaeme@archae.me " } )
2017-04-03 12:28:19 -04:00
mentions = TwitterAPI . parse_mentions ( text )
2017-05-15 12:25:21 -04:00
expected_text = " <a href=' #{ gsimg . ap_id } '>@gsimg</a> According to <a href=' #{ archaeme . ap_id } '>@archaeme</a>, that is @daggsy. Also hello <a href=' #{ archaeme_remote . ap_id } '>@archaeme@archae.me</a> "
2017-04-03 12:28:19 -04:00
2017-05-17 12:00:20 -04:00
assert Utils . add_user_links ( text , mentions ) == expected_text
2017-04-03 12:28:19 -04:00
end
2017-04-13 10:19:07 -04:00
test " it favorites a status, returns the updated status " do
user = insert ( :user )
note_activity = insert ( :note_activity )
activity_user = Repo . get_by! ( User , ap_id : note_activity . data [ " actor " ] )
{ :ok , status } = TwitterAPI . favorite ( user , note_activity )
updated_activity = Activity . get_by_ap_id ( note_activity . data [ " id " ] )
assert status == ActivityRepresenter . to_map ( updated_activity , %{ user : activity_user , for : user } )
end
2017-04-14 11:13:51 -04:00
2017-04-14 12:15:15 -04:00
test " it unfavorites a status, returns the updated status " do
user = insert ( :user )
note_activity = insert ( :note_activity )
activity_user = Repo . get_by! ( User , ap_id : note_activity . data [ " actor " ] )
object = Object . get_by_ap_id ( note_activity . data [ " object " ] [ " id " ] )
2017-04-14 12:27:17 -04:00
{ :ok , _like_activity , _object } = ActivityPub . like ( user , object )
2017-04-14 12:15:15 -04:00
updated_activity = Activity . get_by_ap_id ( note_activity . data [ " id " ] )
assert ActivityRepresenter . to_map ( updated_activity , %{ user : activity_user , for : user } ) [ " fave_num " ] == 1
{ :ok , status } = TwitterAPI . unfavorite ( user , note_activity )
assert status [ " fave_num " ] == 0
end
2017-04-15 07:54:46 -04:00
test " it retweets a status and returns the retweet " do
user = insert ( :user )
note_activity = insert ( :note_activity )
activity_user = Repo . get_by! ( User , ap_id : note_activity . data [ " actor " ] )
{ :ok , status } = TwitterAPI . retweet ( user , note_activity )
updated_activity = Activity . get_by_ap_id ( note_activity . data [ " id " ] )
assert status == ActivityRepresenter . to_map ( updated_activity , %{ user : activity_user , for : user } )
end
2017-04-16 04:25:27 -04:00
test " it registers a new user and returns the user. " do
data = %{
" nickname " = > " lain " ,
" email " = > " lain@wired.jp " ,
" fullname " = > " lain iwakura " ,
" bio " = > " close the world. " ,
" password " = > " bear " ,
" confirm " = > " bear "
}
{ :ok , user } = TwitterAPI . register_user ( data )
fetched_user = Repo . get_by ( User , nickname : " lain " )
assert user == UserRepresenter . to_map ( fetched_user )
end
test " it returns the error on registration problems " do
data = %{
" nickname " = > " lain " ,
" email " = > " lain@wired.jp " ,
" fullname " = > " lain iwakura " ,
" bio " = > " close the world. " ,
" password " = > " bear "
}
{ :error , error_object } = TwitterAPI . register_user ( data )
assert is_binary ( error_object [ :error ] )
refute Repo . get_by ( User , nickname : " lain " )
end
2017-04-30 07:53:26 -04:00
test " it assigns an integer conversation_id " do
note_activity = insert ( :note_activity )
user = User . get_cached_by_ap_id ( note_activity . data [ " actor " ] )
status = ActivityRepresenter . to_map ( note_activity , %{ user : user } )
assert is_number ( status [ " statusnet_conversation_id " ] )
end
2017-04-14 11:13:51 -04:00
setup do
2017-04-17 05:36:17 -04:00
Supervisor . terminate_child ( Pleroma.Supervisor , Cachex )
Supervisor . restart_child ( Pleroma.Supervisor , Cachex )
2017-04-14 11:13:51 -04:00
:ok
end
2017-04-30 07:53:26 -04:00
describe " context_to_conversation_id " do
test " creates a mapping object " do
conversation_id = TwitterAPI . context_to_conversation_id ( " random context " )
object = Object . get_by_ap_id ( " random context " )
assert conversation_id == object . id
end
test " returns an existing mapping for an existing object " do
{ :ok , object } = Object . context_mapping ( " random context " ) |> Repo . insert
conversation_id = TwitterAPI . context_to_conversation_id ( " random context " )
assert conversation_id == object . id
end
end
2017-05-10 12:44:57 -04:00
describe " fetching a user by uri " do
test " fetches a user by uri " do
user = insert ( :user )
{ :ok , represented } = TwitterAPI . get_external_profile ( user , user . ap_id )
assert represented = UserRepresenter . to_map ( user , %{ for : user } )
end
end
2017-03-21 12:53:20 -04:00
end