2018-12-23 13:11:29 -07:00
|
|
|
# Pleroma: A lightweight social networking server
|
|
|
|
# Copyright © 2017-2018 Pleroma Authors <https://pleroma.social/>
|
|
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2017-04-20 19:59:11 -06:00
|
|
|
defmodule Pleroma.Web.WebsubTest do
|
|
|
|
use Pleroma.DataCase
|
2019-08-09 11:08:01 -06:00
|
|
|
use Oban.Testing, repo: Pleroma.Repo
|
2019-03-04 19:52:23 -07:00
|
|
|
|
2019-08-09 11:08:01 -06:00
|
|
|
alias Pleroma.ObanHelpers
|
2019-03-04 19:52:23 -07:00
|
|
|
alias Pleroma.Web.Router.Helpers
|
2017-04-20 19:59:11 -06:00
|
|
|
alias Pleroma.Web.Websub
|
2019-02-10 14:57:38 -07:00
|
|
|
alias Pleroma.Web.Websub.WebsubClientSubscription
|
2019-03-04 19:52:23 -07:00
|
|
|
alias Pleroma.Web.Websub.WebsubServerSubscription
|
2019-08-09 11:08:01 -06:00
|
|
|
alias Pleroma.Workers.Subscriber, as: SubscriberWorker
|
2019-03-04 19:52:23 -07:00
|
|
|
|
2017-04-20 19:59:11 -06:00
|
|
|
import Pleroma.Factory
|
2018-12-02 06:58:38 -07:00
|
|
|
import Tesla.Mock
|
|
|
|
|
|
|
|
setup do
|
2018-12-02 08:17:26 -07:00
|
|
|
mock(fn env -> apply(HttpRequestMock, :request, [env]) end)
|
2018-12-02 06:58:38 -07:00
|
|
|
:ok
|
|
|
|
end
|
2017-04-20 19:59:11 -06:00
|
|
|
|
|
|
|
test "a verification of a request that is accepted" do
|
|
|
|
sub = insert(:websub_subscription)
|
|
|
|
topic = sub.topic
|
|
|
|
|
2018-03-30 07:01:53 -06:00
|
|
|
getter = fn _path, _headers, options ->
|
2017-04-20 19:59:11 -06:00
|
|
|
%{
|
|
|
|
"hub.challenge": challenge,
|
|
|
|
"hub.lease_seconds": seconds,
|
|
|
|
"hub.topic": ^topic,
|
|
|
|
"hub.mode": "subscribe"
|
|
|
|
} = Keyword.get(options, :params)
|
|
|
|
|
2017-04-22 04:07:51 -06:00
|
|
|
assert String.to_integer(seconds) > 0
|
2017-04-20 19:59:11 -06:00
|
|
|
|
2018-03-30 07:01:53 -06:00
|
|
|
{:ok,
|
2018-12-02 06:58:38 -07:00
|
|
|
%Tesla.Env{
|
|
|
|
status: 200,
|
2018-03-30 07:01:53 -06:00
|
|
|
body: challenge
|
|
|
|
}}
|
2017-04-20 19:59:11 -06:00
|
|
|
end
|
|
|
|
|
|
|
|
{:ok, sub} = Websub.verify(sub, getter)
|
|
|
|
assert sub.state == "active"
|
|
|
|
end
|
|
|
|
|
|
|
|
test "a verification of a request that doesn't return 200" do
|
|
|
|
sub = insert(:websub_subscription)
|
|
|
|
|
2018-03-30 07:01:53 -06:00
|
|
|
getter = fn _path, _headers, _options ->
|
|
|
|
{:ok,
|
2018-12-02 06:58:38 -07:00
|
|
|
%Tesla.Env{
|
|
|
|
status: 500,
|
2018-03-30 07:01:53 -06:00
|
|
|
body: ""
|
|
|
|
}}
|
2017-04-20 19:59:11 -06:00
|
|
|
end
|
|
|
|
|
|
|
|
{:error, sub} = Websub.verify(sub, getter)
|
2017-10-05 04:53:16 -06:00
|
|
|
# Keep the current state.
|
|
|
|
assert sub.state == "requested"
|
2017-04-20 19:59:11 -06:00
|
|
|
end
|
2017-04-22 05:44:21 -06:00
|
|
|
|
|
|
|
test "an incoming subscription request" do
|
|
|
|
user = insert(:user)
|
|
|
|
|
|
|
|
data = %{
|
|
|
|
"hub.callback" => "http://example.org/sub",
|
2017-04-22 05:48:10 -06:00
|
|
|
"hub.mode" => "subscribe",
|
2017-04-22 05:44:21 -06:00
|
|
|
"hub.topic" => Pleroma.Web.OStatus.feed_path(user),
|
|
|
|
"hub.secret" => "a random secret",
|
|
|
|
"hub.lease_seconds" => "100"
|
|
|
|
}
|
|
|
|
|
2018-03-30 07:01:53 -06:00
|
|
|
{:ok, subscription} = Websub.incoming_subscription_request(user, data)
|
2017-04-22 05:44:21 -06:00
|
|
|
assert subscription.topic == Pleroma.Web.OStatus.feed_path(user)
|
|
|
|
assert subscription.state == "requested"
|
|
|
|
assert subscription.secret == "a random secret"
|
|
|
|
assert subscription.callback == "http://example.org/sub"
|
|
|
|
end
|
|
|
|
|
|
|
|
test "an incoming subscription request for an existing subscription" do
|
|
|
|
user = insert(:user)
|
2018-03-30 07:01:53 -06:00
|
|
|
|
|
|
|
sub =
|
|
|
|
insert(:websub_subscription, state: "accepted", topic: Pleroma.Web.OStatus.feed_path(user))
|
2017-04-22 05:44:21 -06:00
|
|
|
|
|
|
|
data = %{
|
|
|
|
"hub.callback" => sub.callback,
|
2017-04-22 05:48:10 -06:00
|
|
|
"hub.mode" => "subscribe",
|
2017-04-22 05:44:21 -06:00
|
|
|
"hub.topic" => Pleroma.Web.OStatus.feed_path(user),
|
|
|
|
"hub.secret" => "a random secret",
|
|
|
|
"hub.lease_seconds" => "100"
|
|
|
|
}
|
|
|
|
|
2018-03-30 07:01:53 -06:00
|
|
|
{:ok, subscription} = Websub.incoming_subscription_request(user, data)
|
2017-04-22 05:44:21 -06:00
|
|
|
assert subscription.topic == Pleroma.Web.OStatus.feed_path(user)
|
|
|
|
assert subscription.state == sub.state
|
|
|
|
assert subscription.secret == "a random secret"
|
|
|
|
assert subscription.callback == sub.callback
|
|
|
|
assert length(Repo.all(WebsubServerSubscription)) == 1
|
|
|
|
assert subscription.id == sub.id
|
|
|
|
end
|
2017-04-26 10:33:10 -06:00
|
|
|
|
2017-04-27 01:46:45 -06:00
|
|
|
def accepting_verifier(subscription) do
|
2018-03-30 07:01:53 -06:00
|
|
|
{:ok, %{subscription | state: "accepted"}}
|
2017-04-27 01:46:45 -06:00
|
|
|
end
|
|
|
|
|
2017-04-26 10:33:10 -06:00
|
|
|
test "initiate a subscription for a given user and topic" do
|
2017-04-29 11:06:01 -06:00
|
|
|
subscriber = insert(:user)
|
2018-11-30 09:08:02 -07:00
|
|
|
user = insert(:user, %{info: %Pleroma.User.Info{topic: "some_topic", hub: "some_hub"}})
|
2017-04-26 10:33:10 -06:00
|
|
|
|
2017-04-29 11:06:01 -06:00
|
|
|
{:ok, websub} = Websub.subscribe(subscriber, user, &accepting_verifier/1)
|
|
|
|
assert websub.subscribers == [subscriber.ap_id]
|
|
|
|
assert websub.topic == "some_topic"
|
|
|
|
assert websub.hub == "some_hub"
|
2017-04-26 10:33:10 -06:00
|
|
|
assert is_binary(websub.secret)
|
2017-04-27 01:46:45 -06:00
|
|
|
assert websub.user == user
|
2017-04-26 10:33:10 -06:00
|
|
|
assert websub.state == "accepted"
|
|
|
|
end
|
2017-04-27 01:46:45 -06:00
|
|
|
|
|
|
|
test "discovers the hub and canonical url" do
|
|
|
|
topic = "https://mastodon.social/users/lambadalambda.atom"
|
|
|
|
|
2018-12-02 06:58:38 -07:00
|
|
|
{:ok, discovered} = Websub.gather_feed_data(topic)
|
2018-03-30 07:01:53 -06:00
|
|
|
|
2017-04-29 09:51:59 -06:00
|
|
|
expected = %{
|
2017-05-03 06:26:49 -06:00
|
|
|
"hub" => "https://mastodon.social/api/push",
|
|
|
|
"uri" => "https://mastodon.social/users/lambadalambda",
|
|
|
|
"nickname" => "lambadalambda",
|
|
|
|
"name" => "Critical Value",
|
|
|
|
"host" => "mastodon.social",
|
2017-05-24 09:34:38 -06:00
|
|
|
"bio" => "a cool dude.",
|
2018-03-30 07:01:53 -06:00
|
|
|
"avatar" => %{
|
|
|
|
"type" => "Image",
|
|
|
|
"url" => [
|
|
|
|
%{
|
|
|
|
"href" =>
|
|
|
|
"https://files.mastodon.social/accounts/avatars/000/000/264/original/1429214160519.gif?1492379244",
|
|
|
|
"mediaType" => "image/gif",
|
|
|
|
"type" => "Link"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
2017-04-29 09:51:59 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
assert expected == discovered
|
2017-04-27 01:46:45 -06:00
|
|
|
end
|
|
|
|
|
|
|
|
test "calls the hub, requests topic" do
|
|
|
|
hub = "https://social.heldscal.la/main/push/hub"
|
|
|
|
topic = "https://social.heldscal.la/api/statuses/user_timeline/23211.atom"
|
|
|
|
websub = insert(:websub_client_subscription, %{hub: hub, topic: topic})
|
|
|
|
|
2018-03-30 07:01:53 -06:00
|
|
|
poster = fn ^hub, {:form, data}, _headers ->
|
2017-04-27 01:46:45 -06:00
|
|
|
assert Keyword.get(data, :"hub.mode") == "subscribe"
|
2018-03-30 07:01:53 -06:00
|
|
|
|
|
|
|
assert Keyword.get(data, :"hub.callback") ==
|
|
|
|
Helpers.websub_url(
|
|
|
|
Pleroma.Web.Endpoint,
|
|
|
|
:websub_subscription_confirmation,
|
|
|
|
websub.id
|
|
|
|
)
|
|
|
|
|
2018-12-02 06:58:38 -07:00
|
|
|
{:ok, %{status: 202}}
|
2017-04-27 01:46:45 -06:00
|
|
|
end
|
|
|
|
|
|
|
|
task = Task.async(fn -> Websub.request_subscription(websub, poster) end)
|
|
|
|
|
|
|
|
change = Ecto.Changeset.change(websub, %{state: "accepted"})
|
|
|
|
{:ok, _} = Repo.update(change)
|
|
|
|
|
|
|
|
{:ok, websub} = Task.await(task)
|
|
|
|
|
|
|
|
assert websub.state == "accepted"
|
|
|
|
end
|
|
|
|
|
|
|
|
test "rejects the subscription if it can't be accepted" do
|
|
|
|
hub = "https://social.heldscal.la/main/push/hub"
|
|
|
|
topic = "https://social.heldscal.la/api/statuses/user_timeline/23211.atom"
|
|
|
|
websub = insert(:websub_client_subscription, %{hub: hub, topic: topic})
|
|
|
|
|
2018-03-30 07:01:53 -06:00
|
|
|
poster = fn ^hub, {:form, _data}, _headers ->
|
2018-12-02 07:08:36 -07:00
|
|
|
{:ok, %{status: 202}}
|
2017-04-27 01:46:45 -06:00
|
|
|
end
|
|
|
|
|
|
|
|
{:error, websub} = Websub.request_subscription(websub, poster, 1000)
|
|
|
|
assert websub.state == "rejected"
|
|
|
|
|
|
|
|
websub = insert(:websub_client_subscription, %{hub: hub, topic: topic})
|
2018-03-30 07:01:53 -06:00
|
|
|
|
|
|
|
poster = fn ^hub, {:form, _data}, _headers ->
|
2018-12-02 07:08:36 -07:00
|
|
|
{:ok, %{status: 400}}
|
2017-04-27 01:46:45 -06:00
|
|
|
end
|
|
|
|
|
|
|
|
{:error, websub} = Websub.request_subscription(websub, poster, 1000)
|
|
|
|
assert websub.state == "rejected"
|
|
|
|
end
|
2017-05-01 14:02:07 -06:00
|
|
|
|
|
|
|
test "sign a text" do
|
|
|
|
signed = Websub.sign("secret", "text")
|
2018-03-30 07:01:53 -06:00
|
|
|
assert signed == "B8392C23690CCF871F37EC270BE1582DEC57A503" |> String.downcase()
|
2017-05-01 14:02:07 -06:00
|
|
|
|
2018-02-12 02:13:54 -07:00
|
|
|
_signed = Websub.sign("secret", [["て"], ['す']])
|
2017-05-01 14:02:07 -06:00
|
|
|
end
|
2017-05-10 10:44:06 -06:00
|
|
|
|
|
|
|
describe "renewing subscriptions" do
|
|
|
|
test "it renews subscriptions that have less than a day of time left" do
|
|
|
|
day = 60 * 60 * 24
|
2018-03-30 07:01:53 -06:00
|
|
|
now = NaiveDateTime.utc_now()
|
|
|
|
|
|
|
|
still_good =
|
|
|
|
insert(:websub_client_subscription, %{
|
|
|
|
valid_until: NaiveDateTime.add(now, 2 * day),
|
|
|
|
topic: "http://example.org/still_good",
|
2018-12-02 06:58:38 -07:00
|
|
|
hub: "http://example.org/still_good",
|
2018-03-30 07:01:53 -06:00
|
|
|
state: "accepted"
|
|
|
|
})
|
|
|
|
|
|
|
|
needs_refresh =
|
|
|
|
insert(:websub_client_subscription, %{
|
|
|
|
valid_until: NaiveDateTime.add(now, day - 100),
|
|
|
|
topic: "http://example.org/needs_refresh",
|
2018-12-02 06:58:38 -07:00
|
|
|
hub: "http://example.org/needs_refresh",
|
2018-03-30 07:01:53 -06:00
|
|
|
state: "accepted"
|
|
|
|
})
|
2017-05-10 10:44:06 -06:00
|
|
|
|
2018-02-12 02:13:54 -07:00
|
|
|
_refresh = Websub.refresh_subscriptions()
|
2019-08-09 11:08:01 -06:00
|
|
|
ObanHelpers.perform(all_enqueued(worker: SubscriberWorker))
|
2017-05-10 10:44:06 -06:00
|
|
|
|
|
|
|
assert still_good == Repo.get(WebsubClientSubscription, still_good.id)
|
|
|
|
refute needs_refresh == Repo.get(WebsubClientSubscription, needs_refresh.id)
|
|
|
|
end
|
|
|
|
end
|
2017-04-20 19:59:11 -06:00
|
|
|
end
|