2019-10-07 12:20:41 +00:00
|
|
|
# Pleroma: A lightweight social networking server
|
2022-02-26 06:11:42 +00:00
|
|
|
# Copyright © 2017-2022 Pleroma Authors <https://pleroma.social/>
|
2019-10-07 12:20:41 +00:00
|
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2019-12-06 06:32:29 +00:00
|
|
|
defmodule Pleroma.Web.Feed.UserControllerTest do
|
2019-10-07 12:20:41 +00:00
|
|
|
use Pleroma.Web.ConnCase
|
|
|
|
|
|
|
|
import Pleroma.Factory
|
2019-11-08 06:23:24 +00:00
|
|
|
import SweetXml
|
2019-10-07 12:20:41 +00:00
|
|
|
|
|
|
|
alias Pleroma.Object
|
|
|
|
alias Pleroma.User
|
2020-05-25 20:27:47 +00:00
|
|
|
alias Pleroma.Web.CommonAPI
|
2020-11-10 07:44:22 +00:00
|
|
|
alias Pleroma.Web.Feed.FeedView
|
2019-10-07 12:20:41 +00:00
|
|
|
|
2020-10-02 19:18:02 +00:00
|
|
|
setup do: clear_config([:static_fe, :enabled], false)
|
2019-10-07 12:20:41 +00:00
|
|
|
|
2020-02-22 16:48:41 +00:00
|
|
|
describe "feed" do
|
2020-03-20 15:33:00 +00:00
|
|
|
setup do: clear_config([:feed])
|
2019-10-07 12:20:41 +00:00
|
|
|
|
2020-11-10 07:44:22 +00:00
|
|
|
setup do
|
2021-01-26 17:58:43 +00:00
|
|
|
clear_config(
|
2020-02-22 16:48:41 +00:00
|
|
|
[:feed, :post_title],
|
2020-11-10 07:44:22 +00:00
|
|
|
%{max_length: 15, omission: "..."}
|
2019-11-08 06:23:24 +00:00
|
|
|
)
|
|
|
|
|
2020-02-22 16:48:41 +00:00
|
|
|
activity = insert(:note_activity)
|
|
|
|
|
|
|
|
note =
|
|
|
|
insert(:note,
|
|
|
|
data: %{
|
2020-11-10 07:44:22 +00:00
|
|
|
"content" => "This & this is :moominmamma: note ",
|
|
|
|
"source" => "This & this is :moominmamma: note ",
|
2020-02-22 16:48:41 +00:00
|
|
|
"attachment" => [
|
|
|
|
%{
|
|
|
|
"url" => [
|
|
|
|
%{"mediaType" => "image/png", "href" => "https://pleroma.gov/image.png"}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
],
|
2020-11-10 07:44:22 +00:00
|
|
|
"inReplyTo" => activity.data["id"],
|
|
|
|
"context" => "2hu & as",
|
|
|
|
"summary" => "2hu & as"
|
2020-02-22 16:48:41 +00:00
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
note_activity = insert(:note_activity, note: note)
|
|
|
|
user = User.get_cached_by_ap_id(note_activity.data["actor"])
|
2019-11-08 06:23:24 +00:00
|
|
|
|
2020-02-22 16:48:41 +00:00
|
|
|
note2 =
|
|
|
|
insert(:note,
|
|
|
|
user: user,
|
|
|
|
data: %{
|
2020-11-10 07:44:22 +00:00
|
|
|
"content" => "42 & This is :moominmamma: note ",
|
2020-02-22 16:48:41 +00:00
|
|
|
"inReplyTo" => activity.data["id"]
|
|
|
|
}
|
|
|
|
)
|
2019-11-08 06:23:24 +00:00
|
|
|
|
2020-03-10 15:11:48 +00:00
|
|
|
note_activity2 = insert(:note_activity, note: note2)
|
2023-03-06 22:55:24 +00:00
|
|
|
|
|
|
|
note3 =
|
|
|
|
insert(:note,
|
|
|
|
user: user,
|
|
|
|
data: %{
|
|
|
|
"content" => "This note tests whether HTML entities are truncated properly",
|
|
|
|
"summary" => "Won't, didn't fail",
|
|
|
|
"inReplyTo" => note_activity2.id
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
_note_activity3 = insert(:note_activity, note: note3)
|
2021-01-04 12:38:31 +00:00
|
|
|
object = Object.normalize(note_activity, fetch: false)
|
2019-10-07 12:20:41 +00:00
|
|
|
|
2023-03-06 22:55:24 +00:00
|
|
|
encoded_title = FeedView.activity_title(note3.data)
|
|
|
|
|
|
|
|
[user: user, object: object, max_id: note_activity2.id, encoded_title: encoded_title]
|
2020-11-10 07:44:22 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
test "gets an atom feed", %{conn: conn, user: user, object: object, max_id: max_id} do
|
2020-02-22 16:48:41 +00:00
|
|
|
resp =
|
|
|
|
conn
|
2020-03-13 14:41:26 +00:00
|
|
|
|> put_req_header("accept", "application/atom+xml")
|
2020-02-22 16:48:41 +00:00
|
|
|
|> get(user_feed_path(conn, :feed, user.nickname))
|
|
|
|
|> response(200)
|
2019-10-07 12:20:41 +00:00
|
|
|
|
2020-02-22 16:48:41 +00:00
|
|
|
activity_titles =
|
|
|
|
resp
|
|
|
|
|> SweetXml.parse()
|
|
|
|
|> SweetXml.xpath(~x"//entry/title/text()"l)
|
2019-10-07 12:20:41 +00:00
|
|
|
|
2023-03-06 22:55:24 +00:00
|
|
|
assert activity_titles == ['Won\'t, didn\'...', '2hu', '2hu & as']
|
2020-11-10 07:44:22 +00:00
|
|
|
assert resp =~ FeedView.escape(object.data["content"])
|
|
|
|
assert resp =~ FeedView.escape(object.data["summary"])
|
|
|
|
assert resp =~ FeedView.escape(object.data["context"])
|
2020-03-10 15:11:48 +00:00
|
|
|
|
|
|
|
resp =
|
|
|
|
conn
|
2020-03-13 14:41:26 +00:00
|
|
|
|> put_req_header("accept", "application/atom+xml")
|
2020-11-10 07:44:22 +00:00
|
|
|
|> get("/users/#{user.nickname}/feed", %{"max_id" => max_id})
|
2020-03-10 15:11:48 +00:00
|
|
|
|> response(200)
|
|
|
|
|
|
|
|
activity_titles =
|
|
|
|
resp
|
|
|
|
|> SweetXml.parse()
|
|
|
|
|> SweetXml.xpath(~x"//entry/title/text()"l)
|
|
|
|
|
2022-11-15 18:45:08 +00:00
|
|
|
assert activity_titles == ['2hu & as']
|
2020-02-22 16:48:41 +00:00
|
|
|
end
|
|
|
|
|
2020-11-10 07:44:22 +00:00
|
|
|
test "gets a rss feed", %{conn: conn, user: user, object: object, max_id: max_id} do
|
2020-03-13 14:41:26 +00:00
|
|
|
resp =
|
|
|
|
conn
|
|
|
|
|> put_req_header("accept", "application/rss+xml")
|
|
|
|
|> get("/users/#{user.nickname}/feed.rss")
|
|
|
|
|> response(200)
|
|
|
|
|
|
|
|
activity_titles =
|
|
|
|
resp
|
|
|
|
|> SweetXml.parse()
|
|
|
|
|> SweetXml.xpath(~x"//item/title/text()"l)
|
|
|
|
|
2023-03-06 22:55:24 +00:00
|
|
|
assert activity_titles == ['Won\'t, didn\'...', '2hu', '2hu & as']
|
2020-11-10 07:44:22 +00:00
|
|
|
assert resp =~ FeedView.escape(object.data["content"])
|
|
|
|
assert resp =~ FeedView.escape(object.data["summary"])
|
|
|
|
assert resp =~ FeedView.escape(object.data["context"])
|
2020-03-13 14:41:26 +00:00
|
|
|
|
|
|
|
resp =
|
|
|
|
conn
|
2020-03-13 14:58:14 +00:00
|
|
|
|> put_req_header("accept", "application/rss+xml")
|
2020-11-10 07:44:22 +00:00
|
|
|
|> get("/users/#{user.nickname}/feed.rss", %{"max_id" => max_id})
|
2020-03-13 14:41:26 +00:00
|
|
|
|> response(200)
|
|
|
|
|
|
|
|
activity_titles =
|
|
|
|
resp
|
|
|
|
|> SweetXml.parse()
|
|
|
|
|> SweetXml.xpath(~x"//item/title/text()"l)
|
|
|
|
|
2022-11-15 18:45:08 +00:00
|
|
|
assert activity_titles == ['2hu & as']
|
2020-02-22 16:48:41 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
test "returns 404 for a missing feed", %{conn: conn} do
|
|
|
|
conn =
|
|
|
|
conn
|
2020-03-13 14:41:26 +00:00
|
|
|
|> put_req_header("accept", "application/atom+xml")
|
2020-02-22 16:48:41 +00:00
|
|
|
|> get(user_feed_path(conn, :feed, "nonexisting"))
|
|
|
|
|
|
|
|
assert response(conn, 404)
|
|
|
|
end
|
2020-05-25 20:27:47 +00:00
|
|
|
|
|
|
|
test "returns feed with public and unlisted activities", %{conn: conn} do
|
|
|
|
user = insert(:user)
|
|
|
|
|
|
|
|
{:ok, _} = CommonAPI.post(user, %{status: "public", visibility: "public"})
|
|
|
|
{:ok, _} = CommonAPI.post(user, %{status: "direct", visibility: "direct"})
|
|
|
|
{:ok, _} = CommonAPI.post(user, %{status: "unlisted", visibility: "unlisted"})
|
|
|
|
{:ok, _} = CommonAPI.post(user, %{status: "private", visibility: "private"})
|
|
|
|
|
|
|
|
resp =
|
|
|
|
conn
|
|
|
|
|> put_req_header("accept", "application/atom+xml")
|
|
|
|
|> get(user_feed_path(conn, :feed, user.nickname))
|
|
|
|
|> response(200)
|
|
|
|
|
|
|
|
activity_titles =
|
|
|
|
resp
|
|
|
|
|> SweetXml.parse()
|
|
|
|
|> SweetXml.xpath(~x"//entry/title/text()"l)
|
|
|
|
|> Enum.sort()
|
|
|
|
|
|
|
|
assert activity_titles == ['public', 'unlisted']
|
|
|
|
end
|
2020-07-29 12:02:02 +00:00
|
|
|
|
|
|
|
test "returns 404 when the user is remote", %{conn: conn} do
|
|
|
|
user = insert(:user, local: false)
|
|
|
|
|
|
|
|
{:ok, _} = CommonAPI.post(user, %{status: "test"})
|
|
|
|
|
|
|
|
assert conn
|
|
|
|
|> put_req_header("accept", "application/atom+xml")
|
|
|
|
|> get(user_feed_path(conn, :feed, user.nickname))
|
|
|
|
|> response(404)
|
|
|
|
end
|
2020-10-02 19:18:02 +00:00
|
|
|
|
|
|
|
test "does not require authentication on non-federating instances", %{conn: conn} do
|
|
|
|
clear_config([:instance, :federating], false)
|
|
|
|
user = insert(:user)
|
|
|
|
|
|
|
|
conn
|
|
|
|
|> put_req_header("accept", "application/rss+xml")
|
|
|
|
|> get("/users/#{user.nickname}/feed.rss")
|
|
|
|
|> response(200)
|
|
|
|
end
|
2023-03-06 22:55:24 +00:00
|
|
|
|
|
|
|
test "does not mangle HTML entities midway", %{
|
|
|
|
conn: conn,
|
|
|
|
user: user,
|
|
|
|
object: object,
|
|
|
|
encoded_title: encoded_title
|
|
|
|
} do
|
|
|
|
resp =
|
|
|
|
conn
|
|
|
|
|> put_req_header("accept", "application/atom+xml")
|
|
|
|
|> get(user_feed_path(conn, :feed, user.nickname))
|
|
|
|
|> response(200)
|
|
|
|
|
|
|
|
activity_titles =
|
|
|
|
resp
|
|
|
|
|> SweetXml.parse()
|
|
|
|
|> SweetXml.xpath(~x"//entry/title/text()"l)
|
|
|
|
|
|
|
|
assert activity_titles == ['Won\'t, didn\'...', '2hu', '2hu & as']
|
|
|
|
assert resp =~ FeedView.escape(object.data["content"])
|
|
|
|
assert resp =~ FeedView.escape(object.data["summary"])
|
|
|
|
assert resp =~ FeedView.escape(object.data["context"])
|
|
|
|
assert resp =~ encoded_title
|
|
|
|
end
|
2019-10-07 12:20:41 +00:00
|
|
|
end
|
|
|
|
|
2020-03-09 17:51:44 +00:00
|
|
|
# Note: see ActivityPubControllerTest for JSON format tests
|
2019-10-07 12:20:41 +00:00
|
|
|
describe "feed_redirect" do
|
2020-03-09 17:51:44 +00:00
|
|
|
test "with html format, it redirects to user feed", %{conn: conn} do
|
2019-10-07 12:20:41 +00:00
|
|
|
note_activity = insert(:note_activity)
|
|
|
|
user = User.get_cached_by_ap_id(note_activity.data["actor"])
|
|
|
|
|
|
|
|
response =
|
|
|
|
conn
|
|
|
|
|> get("/users/#{user.nickname}")
|
|
|
|
|> response(200)
|
|
|
|
|
|
|
|
assert response ==
|
2020-06-23 18:10:32 +00:00
|
|
|
Pleroma.Web.Fallback.RedirectController.redirector_with_meta(
|
2019-10-07 12:20:41 +00:00
|
|
|
conn,
|
|
|
|
%{user: user}
|
|
|
|
).resp_body
|
|
|
|
end
|
|
|
|
|
2021-08-17 23:32:32 +00:00
|
|
|
test "with html format, it falls back to frontend when user is remote", %{conn: conn} do
|
|
|
|
user = insert(:user, local: false)
|
|
|
|
|
|
|
|
{:ok, _} = CommonAPI.post(user, %{status: "test"})
|
|
|
|
|
|
|
|
response =
|
|
|
|
conn
|
|
|
|
|> get("/users/#{user.nickname}")
|
|
|
|
|> response(200)
|
|
|
|
|
|
|
|
assert response =~ "</html>"
|
|
|
|
end
|
|
|
|
|
|
|
|
test "with html format, it falls back to frontend when user is not found", %{conn: conn} do
|
2019-10-07 12:20:41 +00:00
|
|
|
response =
|
|
|
|
conn
|
|
|
|
|> get("/users/jimm")
|
2021-08-17 23:32:32 +00:00
|
|
|
|> response(200)
|
2019-10-07 12:20:41 +00:00
|
|
|
|
2021-08-17 23:32:32 +00:00
|
|
|
assert response =~ "</html>"
|
2019-10-07 12:20:41 +00:00
|
|
|
end
|
2020-02-22 16:48:41 +00:00
|
|
|
|
2020-03-09 17:51:44 +00:00
|
|
|
test "with non-html / non-json format, it redirects to user feed in atom format", %{
|
|
|
|
conn: conn
|
|
|
|
} do
|
|
|
|
note_activity = insert(:note_activity)
|
|
|
|
user = User.get_cached_by_ap_id(note_activity.data["actor"])
|
2020-02-22 16:48:41 +00:00
|
|
|
|
2020-03-09 17:51:44 +00:00
|
|
|
conn =
|
|
|
|
conn
|
|
|
|
|> put_req_header("accept", "application/xml")
|
|
|
|
|> get("/users/#{user.nickname}")
|
2020-02-22 16:48:41 +00:00
|
|
|
|
2020-03-09 17:51:44 +00:00
|
|
|
assert conn.status == 302
|
2021-05-31 20:09:11 +00:00
|
|
|
|
|
|
|
assert redirected_to(conn) ==
|
|
|
|
"#{Pleroma.Web.Endpoint.url()}/users/#{user.nickname}/feed.atom"
|
2020-02-22 16:48:41 +00:00
|
|
|
end
|
|
|
|
|
2020-03-09 17:51:44 +00:00
|
|
|
test "with non-html / non-json format, it returns error when user is not found", %{conn: conn} do
|
|
|
|
response =
|
|
|
|
conn
|
|
|
|
|> put_req_header("accept", "application/xml")
|
|
|
|
|> get(user_feed_path(conn, :feed, "jimm"))
|
|
|
|
|> response(404)
|
2020-02-22 16:48:41 +00:00
|
|
|
|
2020-03-09 17:51:44 +00:00
|
|
|
assert response == ~S({"error":"Not found"})
|
2020-02-22 16:48:41 +00:00
|
|
|
end
|
|
|
|
end
|
2020-08-31 20:58:21 +00:00
|
|
|
|
|
|
|
describe "private instance" do
|
|
|
|
setup do: clear_config([:instance, :public])
|
|
|
|
|
|
|
|
test "returns 404 for user feed", %{conn: conn} do
|
2021-01-26 17:58:43 +00:00
|
|
|
clear_config([:instance, :public], false)
|
2020-08-31 20:58:21 +00:00
|
|
|
user = insert(:user)
|
|
|
|
|
|
|
|
{:ok, _} = CommonAPI.post(user, %{status: "test"})
|
|
|
|
|
|
|
|
assert conn
|
|
|
|
|> put_req_header("accept", "application/atom+xml")
|
|
|
|
|> get(user_feed_path(conn, :feed, user.nickname))
|
|
|
|
|> response(404)
|
|
|
|
end
|
|
|
|
end
|
2019-10-07 12:20:41 +00:00
|
|
|
end
|