2018-12-23 20:11:29 +00:00
# Pleroma: A lightweight social networking server
2020-03-02 05:08:45 +00:00
# Copyright © 2017-2020 Pleroma Authors <https://pleroma.social/>
2018-12-23 20:11:29 +00:00
# SPDX-License-Identifier: AGPL-3.0-only
2018-02-15 19:00:06 +00:00
defmodule Pleroma.Web.ActivityPub.TransmogrifierTest do
2020-01-25 07:47:30 +00:00
use Oban.Testing , repo : Pleroma.Repo
2018-02-15 19:00:06 +00:00
use Pleroma.DataCase
2020-01-25 07:47:30 +00:00
2019-03-05 02:52:23 +00:00
alias Pleroma.Activity
2019-03-14 19:04:33 +00:00
alias Pleroma.Object
2019-04-17 15:03:35 +01:00
alias Pleroma.Object.Fetcher
2019-08-13 18:20:26 +01:00
alias Pleroma.Tests.ObanHelpers
2019-03-05 02:52:23 +00:00
alias Pleroma.User
2018-02-15 19:00:06 +00:00
alias Pleroma.Web.ActivityPub.Transmogrifier
2019-10-23 20:27:22 +01:00
alias Pleroma.Web.AdminAPI.AccountView
2019-06-29 18:04:50 +01:00
alias Pleroma.Web.CommonAPI
2018-02-17 19:13:12 +00:00
2019-06-29 18:04:50 +01:00
import Mock
2018-02-17 13:11:20 +00:00
import Pleroma.Factory
2019-06-16 12:42:29 +01:00
import ExUnit.CaptureLog
2018-02-15 19:00:06 +00:00
2018-12-04 11:01:39 +00:00
setup_all do
Tesla.Mock . mock_global ( fn env -> apply ( HttpRequestMock , :request , [ env ] ) end )
2018-12-03 15:53:22 +00:00
:ok
end
2020-03-20 15:33:00 +00:00
setup do : clear_config ( [ :instance , :max_remote_account_fields ] )
2019-08-21 19:24:35 +01:00
2018-02-15 19:00:06 +00:00
describe " handle_incoming " do
2020-05-18 13:48:37 +01:00
test " it works for incoming notices with tag not being an array (kroeg) " do
data = File . read! ( " test/fixtures/kroeg-array-less-emoji.json " ) |> Poison . decode! ( )
{ :ok , % Activity { data : data , local : false } } = Transmogrifier . handle_incoming ( data )
object = Object . normalize ( data [ " object " ] )
assert object . data [ " emoji " ] == %{
" icon_e_smile " = > " https://puckipedia.com/forum/images/smilies/icon_e_smile.png "
}
data = File . read! ( " test/fixtures/kroeg-array-less-hashtag.json " ) |> Poison . decode! ( )
{ :ok , % Activity { data : data , local : false } } = Transmogrifier . handle_incoming ( data )
object = Object . normalize ( data [ " object " ] )
assert " test " in object . data [ " tag " ]
end
test " it works for incoming notices with url not being a string (prismo) " do
data = File . read! ( " test/fixtures/prismo-url-map.json " ) |> Poison . decode! ( )
{ :ok , % Activity { data : data , local : false } } = Transmogrifier . handle_incoming ( data )
object = Object . normalize ( data [ " object " ] )
assert object . data [ " url " ] == " https://prismo.news/posts/83 "
end
test " it cleans up incoming notices which are not really DMs " do
user = insert ( :user )
other_user = insert ( :user )
to = [ user . ap_id , other_user . ap_id ]
data =
File . read! ( " test/fixtures/mastodon-post-activity.json " )
|> Poison . decode! ( )
|> Map . put ( " to " , to )
|> Map . put ( " cc " , [ ] )
object =
data [ " object " ]
|> Map . put ( " to " , to )
|> Map . put ( " cc " , [ ] )
data = Map . put ( data , " object " , object )
{ :ok , % Activity { data : data , local : false } = activity } = Transmogrifier . handle_incoming ( data )
assert data [ " to " ] == [ ]
assert data [ " cc " ] == to
object_data = Object . normalize ( activity ) . data
assert object_data [ " to " ] == [ ]
assert object_data [ " cc " ] == to
end
2018-02-19 16:37:45 +00:00
test " it ignores an incoming notice if we already have it " do
activity = insert ( :note_activity )
2018-03-30 14:01:53 +01:00
data =
File . read! ( " test/fixtures/mastodon-post-activity.json " )
|> Poison . decode! ( )
2019-07-08 17:53:02 +01:00
|> Map . put ( " object " , Object . normalize ( activity ) . data )
2018-02-19 16:37:45 +00:00
{ :ok , returned_activity } = Transmogrifier . handle_incoming ( data )
assert activity == returned_activity
end
2019-11-28 09:44:48 +00:00
@tag capture_log : true
2020-02-15 17:41:38 +00:00
test " it fetches reply-to activities if we don't have them " do
2018-03-30 14:01:53 +01:00
data =
File . read! ( " test/fixtures/mastodon-post-activity.json " )
|> Poison . decode! ( )
2018-02-21 14:22:24 +00:00
2018-03-30 14:01:53 +01:00
object =
data [ " object " ]
|> Map . put ( " inReplyTo " , " https://shitposter.club/notice/2827873 " )
2018-02-21 14:22:24 +00:00
2019-06-29 18:04:50 +01:00
data = Map . put ( data , " object " , object )
2018-02-21 14:22:24 +00:00
{ :ok , returned_activity } = Transmogrifier . handle_incoming ( data )
2019-07-09 19:46:16 +01:00
returned_object = Object . normalize ( returned_activity , false )
2018-02-21 14:22:24 +00:00
2018-03-30 14:01:53 +01:00
assert activity =
2019-01-21 06:14:20 +00:00
Activity . get_create_by_object_ap_id (
2018-03-30 14:01:53 +01:00
" tag:shitposter.club,2017-05-05:noticeId=2827873:objectType=comment "
)
2020-09-08 09:43:57 +01:00
assert returned_object . data [ " inReplyTo " ] ==
" tag:shitposter.club,2017-05-05:noticeId=2827873:objectType=comment "
2018-02-25 09:56:01 +00:00
end
2020-02-15 17:41:38 +00:00
test " it does not fetch reply-to activities beyond max replies depth limit " do
2019-06-29 18:04:50 +01:00
data =
File . read! ( " test/fixtures/mastodon-post-activity.json " )
|> Poison . decode! ( )
object =
data [ " object " ]
|> Map . put ( " inReplyTo " , " https://shitposter.club/notice/2827873 " )
data = Map . put ( data , " object " , object )
with_mock Pleroma.Web.Federator ,
2020-02-15 17:41:38 +00:00
allowed_thread_distance? : fn _ -> false end do
2019-06-29 18:04:50 +01:00
{ :ok , returned_activity } = Transmogrifier . handle_incoming ( data )
2019-07-09 19:46:16 +01:00
returned_object = Object . normalize ( returned_activity , false )
2019-06-29 18:04:50 +01:00
refute Activity . get_create_by_object_ap_id (
" tag:shitposter.club,2017-05-05:noticeId=2827873:objectType=comment "
)
2020-09-08 09:43:57 +01:00
assert returned_object . data [ " inReplyTo " ] == " https://shitposter.club/notice/2827873 "
2019-06-29 18:04:50 +01:00
end
end
2019-06-07 18:40:38 +01:00
test " it does not crash if the object in inReplyTo can't be fetched " do
data =
File . read! ( " test/fixtures/mastodon-post-activity.json " )
|> Poison . decode! ( )
object =
data [ " object " ]
2019-06-07 18:48:25 +01:00
|> Map . put ( " inReplyTo " , " https://404.site/whatever " )
2019-06-07 18:40:38 +01:00
data =
data
|> Map . put ( " object " , object )
2019-06-16 12:42:29 +01:00
assert capture_log ( fn ->
{ :ok , _returned_activity } = Transmogrifier . handle_incoming ( data )
2020-07-13 13:27:25 +01:00
end ) =~ " [warn] Couldn't fetch \" https://404.site/whatever \" , error: nil "
2019-06-07 18:40:38 +01:00
end
2020-08-04 13:15:32 +01:00
test " it does not work for deactivated users " do
data = File . read! ( " test/fixtures/mastodon-post-activity.json " ) |> Poison . decode! ( )
insert ( :user , ap_id : data [ " actor " ] , deactivated : true )
assert { :error , _ } = Transmogrifier . handle_incoming ( data )
end
2018-02-15 19:00:06 +00:00
test " it works for incoming notices " do
2018-03-30 14:01:53 +01:00
data = File . read! ( " test/fixtures/mastodon-post-activity.json " ) |> Poison . decode! ( )
2018-02-15 19:00:06 +00:00
{ :ok , % Activity { data : data , local : false } } = Transmogrifier . handle_incoming ( data )
2018-03-30 14:01:53 +01:00
assert data [ " id " ] ==
" http://mastodon.example.org/users/admin/statuses/99512778738411822/activity "
assert data [ " context " ] ==
" tag:mastodon.example.org,2018-02-12:objectId=20:objectType=Conversation "
2018-02-15 19:00:06 +00:00
assert data [ " to " ] == [ " https://www.w3.org/ns/activitystreams # Public " ]
2018-03-30 14:01:53 +01:00
2018-02-15 19:00:06 +00:00
assert data [ " cc " ] == [
2018-03-30 14:01:53 +01:00
" http://mastodon.example.org/users/admin/followers " ,
" http://localtesting.pleroma.lol/users/lain "
]
2018-02-15 19:00:06 +00:00
assert data [ " actor " ] == " http://mastodon.example.org/users/admin "
2019-07-08 17:53:02 +01:00
object_data = Object . normalize ( data [ " object " ] ) . data
assert object_data [ " id " ] ==
" http://mastodon.example.org/users/admin/statuses/99512778738411822 "
2018-02-15 19:00:06 +00:00
2019-07-08 17:53:02 +01:00
assert object_data [ " to " ] == [ " https://www.w3.org/ns/activitystreams # Public " ]
2018-03-30 14:01:53 +01:00
2019-07-08 17:53:02 +01:00
assert object_data [ " cc " ] == [
2018-03-30 14:01:53 +01:00
" http://mastodon.example.org/users/admin/followers " ,
" http://localtesting.pleroma.lol/users/lain "
]
2019-07-08 17:53:02 +01:00
assert object_data [ " actor " ] == " http://mastodon.example.org/users/admin "
assert object_data [ " attributedTo " ] == " http://mastodon.example.org/users/admin "
2018-03-30 14:01:53 +01:00
2019-07-08 17:53:02 +01:00
assert object_data [ " context " ] ==
2018-03-30 14:01:53 +01:00
" tag:mastodon.example.org,2018-02-12:objectId=20:objectType=Conversation "
2019-07-08 17:53:02 +01:00
assert object_data [ " sensitive " ] == true
2018-04-22 09:01:10 +01:00
2019-07-08 17:53:02 +01:00
user = User . get_cached_by_ap_id ( object_data [ " actor " ] )
2018-04-22 09:01:10 +01:00
2019-10-16 19:59:21 +01:00
assert user . note_count == 1
2018-02-15 19:00:06 +00:00
end
2018-02-17 13:55:44 +00:00
2018-03-24 21:39:37 +00:00
test " it works for incoming notices with hashtags " do
2018-03-30 14:01:53 +01:00
data = File . read! ( " test/fixtures/mastodon-post-activity-hashtag.json " ) |> Poison . decode! ( )
2018-03-24 21:39:37 +00:00
{ :ok , % Activity { data : data , local : false } } = Transmogrifier . handle_incoming ( data )
2018-11-25 22:31:07 +00:00
object = Object . normalize ( data [ " object " ] )
assert Enum . at ( object . data [ " tag " ] , 2 ) == " moo "
2018-03-24 21:39:37 +00:00
end
2019-09-27 13:40:31 +01:00
2018-06-18 23:11:48 +01:00
test " it works for incoming notices with contentMap " do
data =
File . read! ( " test/fixtures/mastodon-post-activity-contentmap.json " ) |> Poison . decode! ( )
{ :ok , % Activity { data : data , local : false } } = Transmogrifier . handle_incoming ( data )
2018-11-25 22:31:07 +00:00
object = Object . normalize ( data [ " object " ] )
2018-06-18 23:11:48 +01:00
2018-11-25 22:31:07 +00:00
assert object . data [ " content " ] ==
2018-06-18 23:11:48 +01:00
" <p><span class= \" h-card \" ><a href= \" http://localtesting.pleroma.lol/users/lain \" class= \" u-url mention \" >@<span>lain</span></a></span></p> "
end
2018-08-14 18:07:01 +01:00
test " it works for incoming notices with to/cc not being an array (kroeg) " do
data = File . read! ( " test/fixtures/kroeg-post-activity.json " ) |> Poison . decode! ( )
{ :ok , % Activity { data : data , local : false } } = Transmogrifier . handle_incoming ( data )
2018-11-25 22:31:07 +00:00
object = Object . normalize ( data [ " object " ] )
2018-08-14 18:07:01 +01:00
2018-11-25 22:31:07 +00:00
assert object . data [ " content " ] ==
2018-08-14 18:07:01 +01:00
" <p>henlo from my Psion netBook</p><p>message sent from my Psion netBook</p> "
end
2019-03-19 17:49:29 +00:00
test " it ensures that as:Public activities make it to their followers collection " do
user = insert ( :user )
data =
File . read! ( " test/fixtures/mastodon-post-activity.json " )
|> Poison . decode! ( )
|> Map . put ( " actor " , user . ap_id )
|> Map . put ( " to " , [ " https://www.w3.org/ns/activitystreams # Public " ] )
|> Map . put ( " cc " , [ ] )
object =
data [ " object " ]
|> Map . put ( " attributedTo " , user . ap_id )
|> Map . put ( " to " , [ " https://www.w3.org/ns/activitystreams # Public " ] )
|> Map . put ( " cc " , [ ] )
2019-07-14 18:49:12 +01:00
|> Map . put ( " id " , user . ap_id <> " /activities/12345678 " )
2019-03-19 17:49:29 +00:00
data = Map . put ( data , " object " , object )
{ :ok , % Activity { data : data , local : false } } = Transmogrifier . handle_incoming ( data )
assert data [ " cc " ] == [ User . ap_followers ( user ) ]
end
2019-03-19 17:53:40 +00:00
test " it ensures that address fields become lists " do
user = insert ( :user )
data =
File . read! ( " test/fixtures/mastodon-post-activity.json " )
|> Poison . decode! ( )
|> Map . put ( " actor " , user . ap_id )
|> Map . put ( " to " , nil )
|> Map . put ( " cc " , nil )
object =
data [ " object " ]
|> Map . put ( " attributedTo " , user . ap_id )
|> Map . put ( " to " , nil )
|> Map . put ( " cc " , nil )
2019-07-14 18:49:12 +01:00
|> Map . put ( " id " , user . ap_id <> " /activities/12345678 " )
2019-03-19 17:53:40 +00:00
data = Map . put ( data , " object " , object )
{ :ok , % Activity { data : data , local : false } } = Transmogrifier . handle_incoming ( data )
assert ! is_nil ( data [ " to " ] )
assert ! is_nil ( data [ " cc " ] )
2018-02-17 20:57:31 +00:00
end
2018-02-25 15:14:25 +00:00
2019-08-10 19:47:40 +01:00
test " it strips internal likes " do
data =
File . read! ( " test/fixtures/mastodon-post-activity.json " )
|> Poison . decode! ( )
likes = %{
" first " = >
" http://mastodon.example.org/objects/dbdbc507-52c8-490d-9b7c-1e1d52e5c132/likes?page=1 " ,
" id " = > " http://mastodon.example.org/objects/dbdbc507-52c8-490d-9b7c-1e1d52e5c132/likes " ,
" totalItems " = > 3 ,
" type " = > " OrderedCollection "
}
object = Map . put ( data [ " object " ] , " likes " , likes )
data = Map . put ( data , " object " , object )
{ :ok , % Activity { object : object } } = Transmogrifier . handle_incoming ( data )
refute Map . has_key? ( object . data , " likes " )
end
2019-09-12 17:59:13 +01:00
test " it strips internal reactions " do
user = insert ( :user )
2020-05-12 20:59:26 +01:00
{ :ok , activity } = CommonAPI . post ( user , %{ status : " # cofe " } )
2020-05-05 11:28:28 +01:00
{ :ok , _ } = CommonAPI . react_with_emoji ( activity . id , user , " 📢 " )
2019-09-12 17:59:13 +01:00
%{ object : object } = Activity . get_by_id_with_object ( activity . id )
assert Map . has_key? ( object . data , " reactions " )
assert Map . has_key? ( object . data , " reaction_count " )
object_data = Transmogrifier . strip_internal_fields ( object . data )
refute Map . has_key? ( object_data , " reactions " )
refute Map . has_key? ( object_data , " reaction_count " )
end
2020-08-11 13:00:21 +01:00
test " it works for incoming unfollows with an existing follow " do
2018-05-18 04:55:00 +01:00
user = insert ( :user )
follow_data =
File . read! ( " test/fixtures/mastodon-follow-activity.json " )
|> Poison . decode! ( )
|> Map . put ( " object " , user . ap_id )
{ :ok , % Activity { data : _ , local : false } } = Transmogrifier . handle_incoming ( follow_data )
data =
File . read! ( " test/fixtures/mastodon-unfollow-activity.json " )
|> Poison . decode! ( )
|> Map . put ( " object " , follow_data )
{ :ok , % Activity { data : data , local : false } } = Transmogrifier . handle_incoming ( data )
assert data [ " type " ] == " Undo "
assert data [ " object " ] [ " type " ] == " Follow "
2018-05-20 02:23:52 +01:00
assert data [ " object " ] [ " object " ] == user . ap_id
2018-05-18 04:55:00 +01:00
assert data [ " actor " ] == " http://mastodon.example.org/users/admin "
2019-04-22 08:20:43 +01:00
refute User . following? ( User . get_cached_by_ap_id ( data [ " actor " ] ) , user )
2018-05-18 04:55:00 +01:00
end
2018-05-20 02:23:52 +01:00
2020-04-29 06:13:10 +01:00
test " skip converting the content when it is nil " do
object_id = " https://peertube.social/videos/watch/278d2b7c-0f38-4aaa-afe6-9ecc0c4a34fe "
{ :ok , object } = Fetcher . fetch_and_contain_remote_object_from_id ( object_id )
result =
Pleroma.Web.ActivityPub.Transmogrifier . fix_object ( Map . merge ( object , %{ " content " = > nil } ) )
assert result [ " content " ] == nil
end
2020-04-28 07:32:43 +01:00
test " it converts content of object to html " do
object_id = " https://peertube.social/videos/watch/278d2b7c-0f38-4aaa-afe6-9ecc0c4a34fe "
{ :ok , %{ " content " = > content_markdown } } =
Fetcher . fetch_and_contain_remote_object_from_id ( object_id )
{ :ok , % Pleroma.Object { data : %{ " content " = > content } } = object } =
Fetcher . fetch_object_from_id ( object_id )
assert content_markdown ==
" Support this and our other Michigan!/usr/group videos and meetings. Learn more at http://mug.org/membership \n \n Twenty Years in Jail: FreeBSD's Jails, Then and Now \n \n Jails started as a limited virtualization system, but over the last two years they've... "
assert content ==
" <p>Support this and our other Michigan!/usr/group videos and meetings. Learn more at <a href= \" http://mug.org/membership \" >http://mug.org/membership</a></p><p>Twenty Years in Jail: FreeBSD’ s Jails, Then and Now</p><p>Jails started as a limited virtualization system, but over the last two years they’ ve…</p> "
2020-04-13 04:53:45 +01:00
assert object . data [ " mediaType " ] == " text/html "
end
2018-12-23 13:28:17 +00:00
test " it remaps video URLs as attachments if necessary " do
{ :ok , object } =
2019-04-17 15:03:35 +01:00
Fetcher . fetch_object_from_id (
2018-12-23 13:28:17 +00:00
" https://peertube.moe/videos/watch/df5f464b-be8d-46fb-ad81-2d4c2d1630e3 "
)
assert object . data [ " url " ] ==
" https://peertube.moe/videos/watch/df5f464b-be8d-46fb-ad81-2d4c2d1630e3 "
2020-05-25 12:13:42 +01:00
assert object . data [ " attachment " ] == [
%{
" type " = > " Link " ,
" mediaType " = > " video/mp4 " ,
" url " = > [
%{
" href " = >
" https://peertube.moe/static/webseed/df5f464b-be8d-46fb-ad81-2d4c2d1630e3-480.mp4 " ,
2020-06-23 04:30:34 +01:00
" mediaType " = > " video/mp4 " ,
" type " = > " Link "
2020-05-25 12:13:42 +01:00
}
]
}
]
{ :ok , object } =
Fetcher . fetch_object_from_id (
" https://framatube.org/videos/watch/6050732a-8a7a-43d4-a6cd-809525a1d206 "
)
assert object . data [ " attachment " ] == [
%{
" type " = > " Link " ,
" mediaType " = > " video/mp4 " ,
" url " = > [
%{
" href " = >
" https://framatube.org/static/webseed/6050732a-8a7a-43d4-a6cd-809525a1d206-1080.mp4 " ,
2020-06-23 04:30:34 +01:00
" mediaType " = > " video/mp4 " ,
" type " = > " Link "
2020-05-25 12:13:42 +01:00
}
]
}
]
assert object . data [ " url " ] ==
" https://framatube.org/videos/watch/6050732a-8a7a-43d4-a6cd-809525a1d206 "
2018-12-23 13:28:17 +00:00
end
2019-03-14 19:04:33 +00:00
test " it accepts Flag activities " do
user = insert ( :user )
other_user = insert ( :user )
2020-05-12 20:59:26 +01:00
{ :ok , activity } = CommonAPI . post ( user , %{ status : " test post " } )
2019-07-08 17:53:02 +01:00
object = Object . normalize ( activity )
2019-03-14 19:04:33 +00:00
2019-10-23 20:27:22 +01:00
note_obj = %{
" type " = > " Note " ,
" id " = > activity . data [ " id " ] ,
" content " = > " test post " ,
" published " = > object . data [ " published " ] ,
2020-07-23 13:08:30 +01:00
" actor " = > AccountView . render ( " show.json " , %{ user : user , skip_visibility_check : true } )
2019-10-23 20:27:22 +01:00
}
2019-03-14 19:04:33 +00:00
message = %{
" @context " = > " https://www.w3.org/ns/activitystreams " ,
" cc " = > [ user . ap_id ] ,
2019-10-27 13:17:37 +00:00
" object " = > [ user . ap_id , activity . data [ " id " ] ] ,
2019-03-14 19:04:33 +00:00
" type " = > " Flag " ,
" content " = > " blocked AND reported!!! " ,
" actor " = > other_user . ap_id
}
assert { :ok , activity } = Transmogrifier . handle_incoming ( message )
2019-10-23 20:27:22 +01:00
assert activity . data [ " object " ] == [ user . ap_id , note_obj ]
2019-03-14 19:04:33 +00:00
assert activity . data [ " content " ] == " blocked AND reported!!! "
assert activity . data [ " actor " ] == other_user . ap_id
assert activity . data [ " cc " ] == [ user . ap_id ]
end
2019-10-23 01:43:31 +01:00
test " it correctly processes messages with non-array to field " do
user = insert ( :user )
message = %{
" @context " = > " https://www.w3.org/ns/activitystreams " ,
" to " = > " https://www.w3.org/ns/activitystreams # Public " ,
" type " = > " Create " ,
" object " = > %{
" content " = > " blah blah blah " ,
" type " = > " Note " ,
" attributedTo " = > user . ap_id ,
" inReplyTo " = > nil
} ,
" actor " = > user . ap_id
}
assert { :ok , activity } = Transmogrifier . handle_incoming ( message )
assert [ " https://www.w3.org/ns/activitystreams # Public " ] == activity . data [ " to " ]
end
test " it correctly processes messages with non-array cc field " do
user = insert ( :user )
message = %{
" @context " = > " https://www.w3.org/ns/activitystreams " ,
" to " = > user . follower_address ,
" cc " = > " https://www.w3.org/ns/activitystreams # Public " ,
" type " = > " Create " ,
" object " = > %{
" content " = > " blah blah blah " ,
" type " = > " Note " ,
" attributedTo " = > user . ap_id ,
" inReplyTo " = > nil
} ,
" actor " = > user . ap_id
}
assert { :ok , activity } = Transmogrifier . handle_incoming ( message )
assert [ " https://www.w3.org/ns/activitystreams # Public " ] == activity . data [ " cc " ]
assert [ user . follower_address ] == activity . data [ " to " ]
end
2019-10-30 11:21:49 +00:00
2020-07-13 20:19:13 +01:00
test " it correctly processes messages with weirdness in address fields " do
user = insert ( :user )
message = %{
" @context " = > " https://www.w3.org/ns/activitystreams " ,
" to " = > [ nil , user . follower_address ] ,
" cc " = > [ " https://www.w3.org/ns/activitystreams # Public " , [ " ¿ " ] ] ,
" type " = > " Create " ,
" object " = > %{
" content " = > " … " ,
" type " = > " Note " ,
" attributedTo " = > user . ap_id ,
" inReplyTo " = > nil
} ,
" actor " = > user . ap_id
}
assert { :ok , activity } = Transmogrifier . handle_incoming ( message )
assert [ " https://www.w3.org/ns/activitystreams # Public " ] == activity . data [ " cc " ]
assert [ user . follower_address ] == activity . data [ " to " ]
end
2019-10-30 11:21:49 +00:00
test " it accepts Move activities " do
old_user = insert ( :user )
new_user = insert ( :user )
message = %{
" @context " = > " https://www.w3.org/ns/activitystreams " ,
" type " = > " Move " ,
" actor " = > old_user . ap_id ,
" object " = > old_user . ap_id ,
" target " = > new_user . ap_id
}
assert :error = Transmogrifier . handle_incoming ( message )
{ :ok , _new_user } = User . update_and_set_cache ( new_user , %{ also_known_as : [ old_user . ap_id ] } )
assert { :ok , % Activity { } = activity } = Transmogrifier . handle_incoming ( message )
assert activity . actor == old_user . ap_id
assert activity . data [ " actor " ] == old_user . ap_id
assert activity . data [ " object " ] == old_user . ap_id
assert activity . data [ " target " ] == new_user . ap_id
assert activity . data [ " type " ] == " Move "
end
2018-02-15 19:00:06 +00:00
end
2018-02-17 13:11:20 +00:00
2020-02-15 17:41:38 +00:00
describe " `handle_incoming/2`, Mastodon format `replies` handling " do
2020-03-20 15:33:00 +00:00
setup do : clear_config ( [ :activitypub , :note_replies_output_limit ] , 5 )
setup do : clear_config ( [ :instance , :federation_incoming_replies_max_depth ] )
2020-02-15 17:41:38 +00:00
setup do
2020-01-25 07:47:30 +00:00
data =
2020-02-10 08:46:16 +00:00
" test/fixtures/mastodon-post-activity.json "
|> File . read! ( )
2020-01-25 07:47:30 +00:00
|> Poison . decode! ( )
2020-02-10 08:46:16 +00:00
items = get_in ( data , [ " object " , " replies " , " first " , " items " ] )
assert length ( items ) > 0
2020-01-25 07:47:30 +00:00
2020-02-15 17:41:38 +00:00
%{ data : data , items : items }
end
test " schedules background fetching of `replies` items if max thread depth limit allows " , %{
data : data ,
items : items
} do
Pleroma.Config . put ( [ :instance , :federation_incoming_replies_max_depth ] , 10 )
2020-01-25 07:47:30 +00:00
{ :ok , _activity } = Transmogrifier . handle_incoming ( data )
for id <- items do
2020-02-15 17:41:38 +00:00
job_args = %{ " op " = > " fetch_remote " , " id " = > id , " depth " = > 1 }
2020-01-25 07:47:30 +00:00
assert_enqueued ( worker : Pleroma.Workers.RemoteFetcherWorker , args : job_args )
end
end
2020-02-15 17:41:38 +00:00
test " does NOT schedule background fetching of `replies` beyond max thread depth limit allows " ,
%{ data : data } do
Pleroma.Config . put ( [ :instance , :federation_incoming_replies_max_depth ] , 0 )
{ :ok , _activity } = Transmogrifier . handle_incoming ( data )
assert all_enqueued ( worker : Pleroma.Workers.RemoteFetcherWorker ) == [ ]
end
end
describe " `handle_incoming/2`, Pleroma format `replies` handling " do
2020-03-20 15:33:00 +00:00
setup do : clear_config ( [ :activitypub , :note_replies_output_limit ] , 5 )
setup do : clear_config ( [ :instance , :federation_incoming_replies_max_depth ] )
2020-02-15 17:41:38 +00:00
setup do
2020-02-10 08:46:16 +00:00
user = insert ( :user )
2020-05-12 20:59:26 +01:00
{ :ok , activity } = CommonAPI . post ( user , %{ status : " post1 " } )
2020-01-25 07:47:30 +00:00
2020-02-10 08:46:16 +00:00
{ :ok , reply1 } =
2020-05-12 20:59:26 +01:00
CommonAPI . post ( user , %{ status : " reply1 " , in_reply_to_status_id : activity . id } )
2020-01-25 07:47:30 +00:00
2020-02-10 08:46:16 +00:00
{ :ok , reply2 } =
2020-05-12 20:59:26 +01:00
CommonAPI . post ( user , %{ status : " reply2 " , in_reply_to_status_id : activity . id } )
2020-02-10 08:46:16 +00:00
replies_uris = Enum . map ( [ reply1 , reply2 ] , fn a -> a . object . data [ " id " ] end )
{ :ok , federation_output } = Transmogrifier . prepare_outgoing ( activity . data )
Repo . delete ( activity . object )
Repo . delete ( activity )
2020-02-15 17:41:38 +00:00
%{ federation_output : federation_output , replies_uris : replies_uris }
end
test " schedules background fetching of `replies` items if max thread depth limit allows " , %{
federation_output : federation_output ,
replies_uris : replies_uris
} do
Pleroma.Config . put ( [ :instance , :federation_incoming_replies_max_depth ] , 1 )
2020-02-10 08:46:16 +00:00
{ :ok , _activity } = Transmogrifier . handle_incoming ( federation_output )
for id <- replies_uris do
2020-02-15 17:41:38 +00:00
job_args = %{ " op " = > " fetch_remote " , " id " = > id , " depth " = > 1 }
2020-01-25 07:47:30 +00:00
assert_enqueued ( worker : Pleroma.Workers.RemoteFetcherWorker , args : job_args )
end
end
2020-02-15 17:41:38 +00:00
test " does NOT schedule background fetching of `replies` beyond max thread depth limit allows " ,
%{ federation_output : federation_output } do
Pleroma.Config . put ( [ :instance , :federation_incoming_replies_max_depth ] , 0 )
{ :ok , _activity } = Transmogrifier . handle_incoming ( federation_output )
assert all_enqueued ( worker : Pleroma.Workers.RemoteFetcherWorker ) == [ ]
end
2020-01-25 07:47:30 +00:00
end
2018-02-17 13:11:20 +00:00
describe " prepare outgoing " do
2019-10-02 11:14:08 +01:00
test " it inlines private announced objects " do
user = insert ( :user )
2020-05-12 20:59:26 +01:00
{ :ok , activity } = CommonAPI . post ( user , %{ status : " hey " , visibility : " private " } )
2019-10-02 11:14:08 +01:00
2020-05-21 12:16:21 +01:00
{ :ok , announce_activity } = CommonAPI . repeat ( activity . id , user )
2019-10-02 11:14:08 +01:00
{ :ok , modified } = Transmogrifier . prepare_outgoing ( announce_activity . data )
assert modified [ " object " ] [ " content " ] == " hey "
assert modified [ " object " ] [ " actor " ] == modified [ " object " ] [ " attributedTo " ]
end
2018-02-17 13:11:20 +00:00
test " it turns mentions into tags " do
user = insert ( :user )
other_user = insert ( :user )
2018-03-30 14:01:53 +01:00
{ :ok , activity } =
2020-05-12 20:59:26 +01:00
CommonAPI . post ( user , %{ status : " hey, @ #{ other_user . nickname } , how are ya? # 2hu " } )
2018-02-17 13:11:20 +00:00
2020-05-25 14:08:43 +01:00
with_mock Pleroma.Notification ,
get_notified_from_activity : fn _ , _ -> [ ] end do
{ :ok , modified } = Transmogrifier . prepare_outgoing ( activity . data )
2018-02-17 13:11:20 +00:00
2020-05-25 14:08:43 +01:00
object = modified [ " object " ]
2018-02-19 09:39:03 +00:00
2020-05-25 14:08:43 +01:00
expected_mention = %{
" href " = > other_user . ap_id ,
" name " = > " @ #{ other_user . nickname } " ,
" type " = > " Mention "
}
2018-02-17 13:11:20 +00:00
2020-05-25 14:08:43 +01:00
expected_tag = %{
" href " = > Pleroma.Web.Endpoint . url ( ) <> " /tags/2hu " ,
" type " = > " Hashtag " ,
" name " = > " # 2hu "
}
refute called ( Pleroma.Notification . get_notified_from_activity ( :_ , :_ ) )
assert Enum . member? ( object [ " tag " ] , expected_tag )
assert Enum . member? ( object [ " tag " ] , expected_mention )
end
2018-02-17 13:11:20 +00:00
end
2018-02-18 13:07:13 +00:00
test " it adds the sensitive property " do
user = insert ( :user )
2020-05-12 20:59:26 +01:00
{ :ok , activity } = CommonAPI . post ( user , %{ status : " # nsfw hey " } )
2018-02-18 13:07:13 +00:00
{ :ok , modified } = Transmogrifier . prepare_outgoing ( activity . data )
assert modified [ " object " ] [ " sensitive " ]
end
2018-02-18 12:58:52 +00:00
test " it adds the json-ld context and the conversation property " do
2018-02-17 13:11:20 +00:00
user = insert ( :user )
2020-05-12 20:59:26 +01:00
{ :ok , activity } = CommonAPI . post ( user , %{ status : " hey " } )
2018-02-17 13:11:20 +00:00
{ :ok , modified } = Transmogrifier . prepare_outgoing ( activity . data )
2018-11-08 15:39:38 +00:00
assert modified [ " @context " ] ==
Pleroma.Web.ActivityPub.Utils . make_json_ld_header ( ) [ " @context " ]
2018-02-18 12:58:52 +00:00
assert modified [ " object " ] [ " conversation " ] == modified [ " context " ]
2018-02-17 13:11:20 +00:00
end
test " it sets the 'attributedTo' property to the actor of the object if it doesn't have one " do
user = insert ( :user )
2020-05-12 20:59:26 +01:00
{ :ok , activity } = CommonAPI . post ( user , %{ status : " hey " } )
2018-02-17 13:11:20 +00:00
{ :ok , modified } = Transmogrifier . prepare_outgoing ( activity . data )
assert modified [ " object " ] [ " actor " ] == modified [ " object " ] [ " attributedTo " ]
end
2018-03-13 17:46:37 +00:00
2018-11-10 12:16:10 +00:00
test " it strips internal hashtag data " do
user = insert ( :user )
2020-05-12 20:59:26 +01:00
{ :ok , activity } = CommonAPI . post ( user , %{ status : " # 2hu " } )
2018-11-10 12:16:10 +00:00
expected_tag = %{
" href " = > Pleroma.Web.Endpoint . url ( ) <> " /tags/2hu " ,
" type " = > " Hashtag " ,
" name " = > " # 2hu "
}
{ :ok , modified } = Transmogrifier . prepare_outgoing ( activity . data )
assert modified [ " object " ] [ " tag " ] == [ expected_tag ]
end
test " it strips internal fields " do
user = insert ( :user )
2020-05-12 20:59:26 +01:00
{ :ok , activity } = CommonAPI . post ( user , %{ status : " # 2hu :firefox: " } )
2018-11-10 12:16:10 +00:00
{ :ok , modified } = Transmogrifier . prepare_outgoing ( activity . data )
assert length ( modified [ " object " ] [ " tag " ] ) == 2
assert is_nil ( modified [ " object " ] [ " emoji " ] )
assert is_nil ( modified [ " object " ] [ " like_count " ] )
assert is_nil ( modified [ " object " ] [ " announcements " ] )
assert is_nil ( modified [ " object " ] [ " announcement_count " ] )
assert is_nil ( modified [ " object " ] [ " context_id " ] )
end
2019-01-12 16:52:30 +00:00
test " it strips internal fields of article " do
activity = insert ( :article_activity )
{ :ok , modified } = Transmogrifier . prepare_outgoing ( activity . data )
2018-11-10 12:16:10 +00:00
assert length ( modified [ " object " ] [ " tag " ] ) == 2
assert is_nil ( modified [ " object " ] [ " emoji " ] )
assert is_nil ( modified [ " object " ] [ " like_count " ] )
assert is_nil ( modified [ " object " ] [ " announcements " ] )
assert is_nil ( modified [ " object " ] [ " announcement_count " ] )
assert is_nil ( modified [ " object " ] [ " context_id " ] )
2019-08-10 19:47:40 +01:00
assert is_nil ( modified [ " object " ] [ " likes " ] )
2019-01-09 08:22:00 +00:00
end
2018-12-23 15:55:07 +00:00
test " the directMessage flag is present " do
user = insert ( :user )
other_user = insert ( :user )
2020-05-12 20:59:26 +01:00
{ :ok , activity } = CommonAPI . post ( user , %{ status : " 2hu :moominmamma: " } )
2018-12-23 15:55:07 +00:00
{ :ok , modified } = Transmogrifier . prepare_outgoing ( activity . data )
assert modified [ " directMessage " ] == false
2020-05-12 20:59:26 +01:00
{ :ok , activity } = CommonAPI . post ( user , %{ status : " @ #{ other_user . nickname } :moominmamma: " } )
2018-12-23 15:55:07 +00:00
{ :ok , modified } = Transmogrifier . prepare_outgoing ( activity . data )
assert modified [ " directMessage " ] == false
{ :ok , activity } =
CommonAPI . post ( user , %{
2020-05-12 20:59:26 +01:00
status : " @ #{ other_user . nickname } :moominmamma: " ,
visibility : " direct "
2018-12-23 15:55:07 +00:00
} )
{ :ok , modified } = Transmogrifier . prepare_outgoing ( activity . data )
assert modified [ " directMessage " ] == true
end
2019-05-14 14:12:47 +01:00
test " it strips BCC field " do
user = insert ( :user )
{ :ok , list } = Pleroma.List . create ( " foo " , user )
2020-05-12 20:59:26 +01:00
{ :ok , activity } = CommonAPI . post ( user , %{ status : " foobar " , visibility : " list: #{ list . id } " } )
2019-05-14 14:12:47 +01:00
{ :ok , modified } = Transmogrifier . prepare_outgoing ( activity . data )
assert is_nil ( modified [ " bcc " ] )
end
2019-09-28 12:57:24 +01:00
test " it can handle Listen activities " do
listen_activity = insert ( :listen )
{ :ok , modified } = Transmogrifier . prepare_outgoing ( listen_activity . data )
assert modified [ " type " ] == " Listen "
2019-09-28 13:12:35 +01:00
user = insert ( :user )
{ :ok , activity } = CommonAPI . listen ( user , %{ " title " = > " lain radio episode 1 " } )
2019-09-30 14:13:25 +01:00
{ :ok , _modified } = Transmogrifier . prepare_outgoing ( activity . data )
2019-09-28 12:57:24 +01:00
end
2018-02-17 13:11:20 +00:00
end
2018-02-21 21:21:40 +00:00
describe " user upgrade " do
test " it upgrades a user to activitypub " do
2018-03-30 14:01:53 +01:00
user =
insert ( :user , %{
nickname : " rye@niu.moe " ,
local : false ,
ap_id : " https://niu.moe/users/rye " ,
follower_address : User . ap_followers ( % User { nickname : " rye@niu.moe " } )
} )
2019-10-10 20:35:32 +01:00
user_two = insert ( :user )
2020-03-28 15:49:03 +00:00
Pleroma.FollowingRelationship . follow ( user_two , user , :follow_accept )
2018-02-21 21:21:40 +00:00
2020-05-12 20:59:26 +01:00
{ :ok , activity } = CommonAPI . post ( user , %{ status : " test " } )
{ :ok , unrelated_activity } = CommonAPI . post ( user_two , %{ status : " test " } )
2018-02-24 09:42:47 +00:00
assert " http://localhost:4001/users/rye@niu.moe/followers " in activity . recipients
2018-02-21 21:21:40 +00:00
2019-04-22 08:20:43 +01:00
user = User . get_cached_by_id ( user . id )
2019-10-16 19:59:21 +01:00
assert user . note_count == 1
2018-02-21 21:21:40 +00:00
{ :ok , user } = Transmogrifier . upgrade_user_from_ap_id ( " https://niu.moe/users/rye " )
2019-08-13 18:20:26 +01:00
ObanHelpers . perform_all ( )
2019-10-16 19:59:21 +01:00
assert user . ap_enabled
assert user . note_count == 1
2018-02-21 21:21:40 +00:00
assert user . follower_address == " https://niu.moe/users/rye/followers "
2019-07-10 14:01:32 +01:00
assert user . following_address == " https://niu.moe/users/rye/following "
2018-02-21 21:21:40 +00:00
2019-04-22 08:20:43 +01:00
user = User . get_cached_by_id ( user . id )
2019-10-16 19:59:21 +01:00
assert user . note_count == 1
2018-02-21 21:21:40 +00:00
2019-04-02 11:08:03 +01:00
activity = Activity . get_by_id ( activity . id )
2018-02-21 21:21:40 +00:00
assert user . follower_address in activity . recipients
2018-03-30 14:01:53 +01:00
assert %{
" url " = > [
%{
" href " = >
" https://cdn.niu.moe/accounts/avatars/000/033/323/original/fd7f8ae0b3ffedc9.jpeg "
}
]
} = user . avatar
assert %{
" url " = > [
%{
" href " = >
" https://cdn.niu.moe/accounts/headers/000/033/323/original/850b3448fa5fd477.png "
}
]
2019-10-16 19:59:21 +01:00
} = user . banner
2018-03-30 14:01:53 +01:00
2018-02-21 21:21:40 +00:00
refute " ... " in activity . recipients
2019-04-02 11:08:03 +01:00
unrelated_activity = Activity . get_by_id ( unrelated_activity . id )
2018-02-21 21:21:40 +00:00
refute user . follower_address in unrelated_activity . recipients
2019-04-22 08:20:43 +01:00
user_two = User . get_cached_by_id ( user_two . id )
2019-10-10 20:35:32 +01:00
assert User . following? ( user_two , user )
refute " ... " in User . following ( user_two )
2018-02-24 16:36:02 +00:00
end
end
2018-05-19 09:06:23 +01:00
describe " actor rewriting " do
test " it fixes the actor URL property to be a proper URI " do
data = %{
" url " = > %{ " href " = > " http://example.com " }
}
rewritten = Transmogrifier . maybe_fix_user_object ( data )
assert rewritten [ " url " ] == " http://example.com "
end
end
2018-09-02 00:33:10 +01:00
describe " actor origin containment " do
test " it rejects activities which reference objects with bogus origins " do
data = %{
" @context " = > " https://www.w3.org/ns/activitystreams " ,
2018-11-17 18:16:55 +00:00
" id " = > " http://mastodon.example.org/users/admin/activities/1234 " ,
" actor " = > " http://mastodon.example.org/users/admin " ,
2018-09-02 00:33:10 +01:00
" to " = > [ " https://www.w3.org/ns/activitystreams # Public " ] ,
" object " = > " https://info.pleroma.site/activity.json " ,
" type " = > " Announce "
}
2019-10-28 16:51:58 +00:00
assert capture_log ( fn ->
2020-05-18 14:47:26 +01:00
{ :error , _ } = Transmogrifier . handle_incoming ( data )
2019-10-28 16:51:58 +00:00
end ) =~ " Object containment failed "
2018-09-02 00:33:10 +01:00
end
2018-11-17 18:12:11 +00:00
2018-11-17 18:24:58 +00:00
test " it rejects activities which reference objects that have an incorrect attribution (variant 1) " do
2018-11-17 18:12:11 +00:00
data = %{
" @context " = > " https://www.w3.org/ns/activitystreams " ,
" id " = > " http://mastodon.example.org/users/admin/activities/1234 " ,
" actor " = > " http://mastodon.example.org/users/admin " ,
" to " = > [ " https://www.w3.org/ns/activitystreams # Public " ] ,
" object " = > " https://info.pleroma.site/activity2.json " ,
" type " = > " Announce "
}
2019-10-28 16:51:58 +00:00
assert capture_log ( fn ->
2020-05-18 14:47:26 +01:00
{ :error , _ } = Transmogrifier . handle_incoming ( data )
2019-10-28 16:51:58 +00:00
end ) =~ " Object containment failed "
2018-11-17 18:12:11 +00:00
end
2018-11-17 18:24:58 +00:00
test " it rejects activities which reference objects that have an incorrect attribution (variant 2) " do
data = %{
" @context " = > " https://www.w3.org/ns/activitystreams " ,
" id " = > " http://mastodon.example.org/users/admin/activities/1234 " ,
" actor " = > " http://mastodon.example.org/users/admin " ,
" to " = > [ " https://www.w3.org/ns/activitystreams # Public " ] ,
" object " = > " https://info.pleroma.site/activity3.json " ,
" type " = > " Announce "
}
2019-10-28 16:51:58 +00:00
assert capture_log ( fn ->
2020-05-18 14:47:26 +01:00
{ :error , _ } = Transmogrifier . handle_incoming ( data )
2019-10-28 16:51:58 +00:00
end ) =~ " Object containment failed "
2018-11-17 18:24:58 +00:00
end
2018-09-02 00:33:10 +01:00
end
2018-11-17 20:07:49 +00:00
2019-02-14 00:59:18 +00:00
describe " reserialization " do
test " successfully reserializes a message with inReplyTo == nil " do
user = insert ( :user )
message = %{
" @context " = > " https://www.w3.org/ns/activitystreams " ,
" to " = > [ " https://www.w3.org/ns/activitystreams # Public " ] ,
" cc " = > [ ] ,
" type " = > " Create " ,
" object " = > %{
" to " = > [ " https://www.w3.org/ns/activitystreams # Public " ] ,
" cc " = > [ ] ,
" type " = > " Note " ,
" content " = > " Hi " ,
" inReplyTo " = > nil ,
" attributedTo " = > user . ap_id
} ,
" actor " = > user . ap_id
}
{ :ok , activity } = Transmogrifier . handle_incoming ( message )
{ :ok , _ } = Transmogrifier . prepare_outgoing ( activity . data )
end
test " successfully reserializes a message with AS2 objects in IR " do
user = insert ( :user )
message = %{
" @context " = > " https://www.w3.org/ns/activitystreams " ,
" to " = > [ " https://www.w3.org/ns/activitystreams # Public " ] ,
" cc " = > [ ] ,
" type " = > " Create " ,
" object " = > %{
" to " = > [ " https://www.w3.org/ns/activitystreams # Public " ] ,
" cc " = > [ ] ,
" type " = > " Note " ,
" content " = > " Hi " ,
" inReplyTo " = > nil ,
" attributedTo " = > user . ap_id ,
" tag " = > [
%{ " name " = > " # 2hu " , " href " = > " http://example.com/2hu " , " type " = > " Hashtag " } ,
%{ " name " = > " Bob " , " href " = > " http://example.com/bob " , " type " = > " Mention " }
]
} ,
" actor " = > user . ap_id
}
{ :ok , activity } = Transmogrifier . handle_incoming ( message )
{ :ok , _ } = Transmogrifier . prepare_outgoing ( activity . data )
end
end
2019-05-22 19:17:57 +01:00
2019-05-31 12:17:05 +01:00
describe " fix_explicit_addressing " do
2019-06-01 04:26:45 +01:00
setup do
2019-05-31 12:17:05 +01:00
user = insert ( :user )
2019-06-01 04:26:45 +01:00
[ user : user ]
end
2019-05-31 12:17:05 +01:00
2019-06-01 04:26:45 +01:00
test " moves non-explicitly mentioned actors to cc " , %{ user : user } do
2019-05-31 12:17:05 +01:00
explicitly_mentioned_actors = [
" https://pleroma.gold/users/user1 " ,
" https://pleroma.gold/user2 "
]
object = %{
" actor " = > user . ap_id ,
" to " = > explicitly_mentioned_actors ++ [ " https://social.beepboop.ga/users/dirb " ] ,
" cc " = > [ ] ,
" tag " = >
Enum . map ( explicitly_mentioned_actors , fn href ->
%{ " type " = > " Mention " , " href " = > href }
end )
}
fixed_object = Transmogrifier . fix_explicit_addressing ( object )
assert Enum . all? ( explicitly_mentioned_actors , & ( &1 in fixed_object [ " to " ] ) )
refute " https://social.beepboop.ga/users/dirb " in fixed_object [ " to " ]
assert " https://social.beepboop.ga/users/dirb " in fixed_object [ " cc " ]
end
2019-06-01 04:26:45 +01:00
test " does not move actor's follower collection to cc " , %{ user : user } do
2019-05-31 12:17:05 +01:00
object = %{
" actor " = > user . ap_id ,
" to " = > [ user . follower_address ] ,
" cc " = > [ ]
}
fixed_object = Transmogrifier . fix_explicit_addressing ( object )
assert user . follower_address in fixed_object [ " to " ]
refute user . follower_address in fixed_object [ " cc " ]
end
2019-06-01 04:26:45 +01:00
test " removes recipient's follower collection from cc " , %{ user : user } do
recipient = insert ( :user )
object = %{
" actor " = > user . ap_id ,
" to " = > [ recipient . ap_id , " https://www.w3.org/ns/activitystreams # Public " ] ,
" cc " = > [ user . follower_address , recipient . follower_address ]
}
fixed_object = Transmogrifier . fix_explicit_addressing ( object )
assert user . follower_address in fixed_object [ " cc " ]
refute recipient . follower_address in fixed_object [ " cc " ]
refute recipient . follower_address in fixed_object [ " to " ]
end
2019-05-31 12:17:05 +01:00
end
2019-09-10 14:43:10 +01:00
describe " fix_summary/1 " do
test " returns fixed object " do
assert Transmogrifier . fix_summary ( %{ " summary " = > nil } ) == %{ " summary " = > " " }
assert Transmogrifier . fix_summary ( %{ " summary " = > " ok " } ) == %{ " summary " = > " ok " }
assert Transmogrifier . fix_summary ( %{ } ) == %{ " summary " = > " " }
end
end
describe " fix_in_reply_to/2 " do
2020-03-20 15:33:00 +00:00
setup do : clear_config ( [ :instance , :federation_incoming_replies_max_depth ] )
2019-09-10 14:43:10 +01:00
setup do
data = Poison . decode! ( File . read! ( " test/fixtures/mastodon-post-activity.json " ) )
[ data : data ]
end
test " returns not modified object when hasn't containts inReplyTo field " , %{ data : data } do
assert Transmogrifier . fix_in_reply_to ( data ) == data
end
2020-09-08 09:43:57 +01:00
test " returns object with inReplyTo when denied incoming reply " , %{ data : data } do
2019-09-10 14:43:10 +01:00
Pleroma.Config . put ( [ :instance , :federation_incoming_replies_max_depth ] , 0 )
object_with_reply =
Map . put ( data [ " object " ] , " inReplyTo " , " https://shitposter.club/notice/2827873 " )
modified_object = Transmogrifier . fix_in_reply_to ( object_with_reply )
assert modified_object [ " inReplyTo " ] == " https://shitposter.club/notice/2827873 "
object_with_reply =
Map . put ( data [ " object " ] , " inReplyTo " , %{ " id " = > " https://shitposter.club/notice/2827873 " } )
modified_object = Transmogrifier . fix_in_reply_to ( object_with_reply )
assert modified_object [ " inReplyTo " ] == %{ " id " = > " https://shitposter.club/notice/2827873 " }
object_with_reply =
Map . put ( data [ " object " ] , " inReplyTo " , [ " https://shitposter.club/notice/2827873 " ] )
modified_object = Transmogrifier . fix_in_reply_to ( object_with_reply )
assert modified_object [ " inReplyTo " ] == [ " https://shitposter.club/notice/2827873 " ]
object_with_reply = Map . put ( data [ " object " ] , " inReplyTo " , [ ] )
modified_object = Transmogrifier . fix_in_reply_to ( object_with_reply )
assert modified_object [ " inReplyTo " ] == [ ]
end
2019-11-28 09:44:48 +00:00
@tag capture_log : true
2019-09-10 14:43:10 +01:00
test " returns modified object when allowed incoming reply " , %{ data : data } do
object_with_reply =
Map . put (
data [ " object " ] ,
" inReplyTo " ,
" https://shitposter.club/notice/2827873 "
)
Pleroma.Config . put ( [ :instance , :federation_incoming_replies_max_depth ] , 5 )
modified_object = Transmogrifier . fix_in_reply_to ( object_with_reply )
assert modified_object [ " inReplyTo " ] ==
" tag:shitposter.club,2017-05-05:noticeId=2827873:objectType=comment "
assert modified_object [ " context " ] ==
" tag:shitposter.club,2017-05-05:objectType=thread:nonce=3c16e9c2681f6d26 "
end
end
describe " fix_url/1 " do
test " fixes data for object when url is map " do
object = %{
" url " = > %{
" type " = > " Link " ,
" mimeType " = > " video/mp4 " ,
" href " = > " https://peede8d-46fb-ad81-2d4c2d1630e3-480.mp4 "
}
}
assert Transmogrifier . fix_url ( object ) == %{
" url " = > " https://peede8d-46fb-ad81-2d4c2d1630e3-480.mp4 "
}
end
test " fixes data for video object " do
object = %{
" type " = > " Video " ,
" url " = > [
%{
" type " = > " Link " ,
" mimeType " = > " video/mp4 " ,
" href " = > " https://peede8d-46fb-ad81-2d4c2d1630e3-480.mp4 "
} ,
%{
" type " = > " Link " ,
" mimeType " = > " video/mp4 " ,
" href " = > " https://peertube46fb-ad81-2d4c2d1630e3-240.mp4 "
} ,
%{
" type " = > " Link " ,
" mimeType " = > " text/html " ,
" href " = > " https://peertube.-2d4c2d1630e3 "
} ,
%{
" type " = > " Link " ,
" mimeType " = > " text/html " ,
" href " = > " https://peertube.-2d4c2d16377-42 "
}
]
}
assert Transmogrifier . fix_url ( object ) == %{
" attachment " = > [
%{
" href " = > " https://peede8d-46fb-ad81-2d4c2d1630e3-480.mp4 " ,
" mimeType " = > " video/mp4 " ,
" type " = > " Link "
}
] ,
" type " = > " Video " ,
" url " = > " https://peertube.-2d4c2d1630e3 "
}
end
test " fixes url for not Video object " do
object = %{
" type " = > " Text " ,
" url " = > [
%{
" type " = > " Link " ,
" mimeType " = > " text/html " ,
" href " = > " https://peertube.-2d4c2d1630e3 "
} ,
%{
" type " = > " Link " ,
" mimeType " = > " text/html " ,
" href " = > " https://peertube.-2d4c2d16377-42 "
}
]
}
assert Transmogrifier . fix_url ( object ) == %{
" type " = > " Text " ,
" url " = > " https://peertube.-2d4c2d1630e3 "
}
assert Transmogrifier . fix_url ( %{ " type " = > " Text " , " url " = > [ ] } ) == %{
" type " = > " Text " ,
" url " = > " "
}
end
test " retunrs not modified object " do
assert Transmogrifier . fix_url ( %{ " type " = > " Text " } ) == %{ " type " = > " Text " }
end
end
2019-09-11 05:23:33 +01:00
describe " get_obj_helper/2 " do
test " returns nil when cannot normalize object " do
2019-10-28 16:51:58 +00:00
assert capture_log ( fn ->
refute Transmogrifier . get_obj_helper ( " test-obj-id " )
end ) =~ " Unsupported URI scheme "
2019-09-11 05:23:33 +01:00
end
2019-11-28 09:44:48 +00:00
@tag capture_log : true
2019-09-11 05:23:33 +01:00
test " returns {:ok, %Object{}} for success case " do
assert { :ok , % Object { } } =
Transmogrifier . get_obj_helper ( " https://shitposter.club/notice/2827873 " )
end
end
describe " fix_attachments/1 " do
test " returns not modified object " do
data = Poison . decode! ( File . read! ( " test/fixtures/mastodon-post-activity.json " ) )
assert Transmogrifier . fix_attachments ( data ) == data
end
test " returns modified object when attachment is map " do
assert Transmogrifier . fix_attachments ( %{
" attachment " = > %{
" mediaType " = > " video/mp4 " ,
" url " = > " https://peertube.moe/stat-480.mp4 "
}
} ) == %{
" attachment " = > [
%{
" mediaType " = > " video/mp4 " ,
2020-06-23 04:30:34 +01:00
" type " = > " Document " ,
2019-09-11 05:23:33 +01:00
" url " = > [
2020-06-23 04:30:34 +01:00
%{
" href " = > " https://peertube.moe/stat-480.mp4 " ,
" mediaType " = > " video/mp4 " ,
" type " = > " Link "
}
2019-09-11 05:23:33 +01:00
]
}
]
}
end
test " returns modified object when attachment is list " do
assert Transmogrifier . fix_attachments ( %{
" attachment " = > [
%{ " mediaType " = > " video/mp4 " , " url " = > " https://pe.er/stat-480.mp4 " } ,
%{ " mimeType " = > " video/mp4 " , " href " = > " https://pe.er/stat-480.mp4 " }
]
} ) == %{
" attachment " = > [
%{
" mediaType " = > " video/mp4 " ,
2020-06-23 04:30:34 +01:00
" type " = > " Document " ,
2019-09-11 05:23:33 +01:00
" url " = > [
2020-06-23 04:30:34 +01:00
%{
" href " = > " https://pe.er/stat-480.mp4 " ,
" mediaType " = > " video/mp4 " ,
" type " = > " Link "
}
2019-09-11 05:23:33 +01:00
]
} ,
%{
" mediaType " = > " video/mp4 " ,
2020-06-23 04:30:34 +01:00
" type " = > " Document " ,
2019-09-11 05:23:33 +01:00
" url " = > [
2020-06-23 04:30:34 +01:00
%{
" href " = > " https://pe.er/stat-480.mp4 " ,
" mediaType " = > " video/mp4 " ,
" type " = > " Link "
}
2019-09-11 05:23:33 +01:00
]
}
]
}
end
end
2019-09-11 21:19:06 +01:00
describe " fix_emoji/1 " do
test " returns not modified object when object not contains tags " do
data = Poison . decode! ( File . read! ( " test/fixtures/mastodon-post-activity.json " ) )
assert Transmogrifier . fix_emoji ( data ) == data
end
test " returns object with emoji when object contains list tags " do
assert Transmogrifier . fix_emoji ( %{
" tag " = > [
%{ " type " = > " Emoji " , " name " = > " :bib: " , " icon " = > %{ " url " = > " /test " } } ,
%{ " type " = > " Hashtag " }
]
} ) == %{
" emoji " = > %{ " bib " = > " /test " } ,
" tag " = > [
%{ " icon " = > %{ " url " = > " /test " } , " name " = > " :bib: " , " type " = > " Emoji " } ,
%{ " type " = > " Hashtag " }
]
}
end
test " returns object with emoji when object contains map tag " do
assert Transmogrifier . fix_emoji ( %{
" tag " = > %{ " type " = > " Emoji " , " name " = > " :bib: " , " icon " = > %{ " url " = > " /test " } }
} ) == %{
" emoji " = > %{ " bib " = > " /test " } ,
" tag " = > %{ " icon " = > %{ " url " = > " /test " } , " name " = > " :bib: " , " type " = > " Emoji " }
}
end
end
2020-01-22 18:10:17 +00:00
describe " set_replies/1 " do
2020-03-20 15:33:00 +00:00
setup do : clear_config ( [ :activitypub , :note_replies_output_limit ] , 2 )
2020-01-22 18:10:17 +00:00
test " returns unmodified object if activity doesn't have self-replies " do
data = Poison . decode! ( File . read! ( " test/fixtures/mastodon-post-activity.json " ) )
assert Transmogrifier . set_replies ( data ) == data
end
test " sets `replies` collection with a limited number of self-replies " do
[ user , another_user ] = insert_list ( 2 , :user )
2020-05-12 20:59:26 +01:00
{ :ok , %{ id : id1 } = activity } = CommonAPI . post ( user , %{ status : " 1 " } )
2020-01-22 18:10:17 +00:00
{ :ok , %{ id : id2 } = self_reply1 } =
2020-05-12 20:59:26 +01:00
CommonAPI . post ( user , %{ status : " self-reply 1 " , in_reply_to_status_id : id1 } )
2020-01-22 18:10:17 +00:00
{ :ok , self_reply2 } =
2020-05-12 20:59:26 +01:00
CommonAPI . post ( user , %{ status : " self-reply 2 " , in_reply_to_status_id : id1 } )
2020-01-22 18:10:17 +00:00
2020-02-08 16:58:02 +00:00
# Assuming to _not_ be present in `replies` due to :note_replies_output_limit is set to 2
2020-05-12 20:59:26 +01:00
{ :ok , _ } = CommonAPI . post ( user , %{ status : " self-reply 3 " , in_reply_to_status_id : id1 } )
2020-01-22 18:10:17 +00:00
{ :ok , _ } =
CommonAPI . post ( user , %{
2020-05-12 20:59:26 +01:00
status : " self-reply to self-reply " ,
in_reply_to_status_id : id2
2020-01-22 18:10:17 +00:00
} )
{ :ok , _ } =
CommonAPI . post ( another_user , %{
2020-05-12 20:59:26 +01:00
status : " another user's reply " ,
in_reply_to_status_id : id1
2020-01-22 18:10:17 +00:00
} )
object = Object . normalize ( activity )
2020-02-09 07:17:21 +00:00
replies_uris = Enum . map ( [ self_reply1 , self_reply2 ] , fn a -> a . object . data [ " id " ] end )
2020-01-22 18:10:17 +00:00
2020-02-09 14:34:48 +00:00
assert %{ " type " = > " Collection " , " items " = > ^ replies_uris } =
Transmogrifier . set_replies ( object . data ) [ " replies " ]
2020-01-22 18:10:17 +00:00
end
end
2020-04-03 12:03:32 +01:00
test " take_emoji_tags/1 " do
user = insert ( :user , %{ emoji : %{ " firefox " = > " https://example.org/firefox.png " } } )
assert Transmogrifier . take_emoji_tags ( user ) == [
%{
" icon " = > %{ " type " = > " Image " , " url " = > " https://example.org/firefox.png " } ,
" id " = > " https://example.org/firefox.png " ,
" name " = > " :firefox: " ,
" type " = > " Emoji " ,
" updated " = > " 1970-01-01T00:00:00Z "
}
]
end
2018-02-15 19:00:06 +00:00
end