2022-06-27 20:55:56 +02:00
|
|
|
from unittest import mock
|
|
|
|
|
2022-06-22 20:11:22 +02:00
|
|
|
import respx
|
|
|
|
from fastapi.testclient import TestClient
|
2022-06-29 20:43:17 +02:00
|
|
|
from sqlalchemy.orm import Session
|
2022-06-22 20:11:22 +02:00
|
|
|
|
2022-06-27 20:55:56 +02:00
|
|
|
from app import activitypub as ap
|
2022-06-22 20:11:22 +02:00
|
|
|
from app import models
|
2022-06-27 20:55:56 +02:00
|
|
|
from app import webfinger
|
2022-06-22 20:11:22 +02:00
|
|
|
from app.config import generate_csrf_token
|
|
|
|
from tests.utils import generate_admin_session_cookies
|
2022-07-26 20:26:34 +02:00
|
|
|
from tests.utils import setup_remote_actor
|
|
|
|
from tests.utils import setup_remote_actor_as_follower
|
2022-06-22 20:11:22 +02:00
|
|
|
|
|
|
|
|
2022-07-01 19:35:34 +02:00
|
|
|
def test_outbox__no_activities(
|
|
|
|
db: Session,
|
|
|
|
client: TestClient,
|
|
|
|
) -> None:
|
|
|
|
response = client.get("/outbox", headers={"Accept": ap.AP_CONTENT_TYPE})
|
|
|
|
|
|
|
|
assert response.status_code == 200
|
|
|
|
|
|
|
|
json_response = response.json()
|
|
|
|
assert json_response["totalItems"] == 0
|
|
|
|
assert json_response["orderedItems"] == []
|
|
|
|
|
|
|
|
|
2022-07-26 19:06:20 +02:00
|
|
|
def test_send_follow_request(
|
|
|
|
db: Session,
|
|
|
|
client: TestClient,
|
|
|
|
respx_mock: respx.MockRouter,
|
|
|
|
) -> None:
|
|
|
|
# given a remote actor
|
2022-07-26 20:26:34 +02:00
|
|
|
ra = setup_remote_actor(respx_mock)
|
2022-06-22 20:11:22 +02:00
|
|
|
|
|
|
|
response = client.post(
|
|
|
|
"/admin/actions/follow",
|
|
|
|
data={
|
|
|
|
"redirect_url": "http://testserver/",
|
|
|
|
"ap_actor_id": ra.ap_id,
|
|
|
|
"csrf_token": generate_csrf_token(),
|
|
|
|
},
|
|
|
|
cookies=generate_admin_session_cookies(),
|
|
|
|
)
|
|
|
|
|
|
|
|
# Then the server returns a 302
|
|
|
|
assert response.status_code == 302
|
|
|
|
assert response.headers.get("Location") == "http://testserver/"
|
|
|
|
|
|
|
|
# And the Follow activity was created in the outbox
|
|
|
|
outbox_object = db.query(models.OutboxObject).one()
|
|
|
|
assert outbox_object.ap_type == "Follow"
|
|
|
|
assert outbox_object.activity_object_ap_id == ra.ap_id
|
|
|
|
|
|
|
|
# And an outgoing activity was queued
|
|
|
|
outgoing_activity = db.query(models.OutgoingActivity).one()
|
|
|
|
assert outgoing_activity.outbox_object_id == outbox_object.id
|
2022-06-27 20:55:56 +02:00
|
|
|
assert outgoing_activity.recipient == ra.inbox_url
|
|
|
|
|
|
|
|
|
|
|
|
def test_send_create_activity__no_followers_and_with_mention(
|
|
|
|
db: Session,
|
|
|
|
client: TestClient,
|
|
|
|
respx_mock: respx.MockRouter,
|
|
|
|
) -> None:
|
|
|
|
# given a remote actor
|
2022-07-26 20:26:34 +02:00
|
|
|
ra = setup_remote_actor(respx_mock)
|
2022-06-27 20:55:56 +02:00
|
|
|
|
|
|
|
with mock.patch.object(webfinger, "get_actor_url", return_value=ra.ap_id):
|
|
|
|
response = client.post(
|
|
|
|
"/admin/actions/new",
|
|
|
|
data={
|
|
|
|
"redirect_url": "http://testserver/",
|
|
|
|
"content": "hi @toto@example.com",
|
|
|
|
"visibility": ap.VisibilityEnum.PUBLIC.name,
|
|
|
|
"csrf_token": generate_csrf_token(),
|
|
|
|
},
|
|
|
|
cookies=generate_admin_session_cookies(),
|
|
|
|
)
|
|
|
|
|
|
|
|
# Then the server returns a 302
|
|
|
|
assert response.status_code == 302
|
|
|
|
|
|
|
|
# And the Follow activity was created in the outbox
|
|
|
|
outbox_object = db.query(models.OutboxObject).one()
|
|
|
|
assert outbox_object.ap_type == "Note"
|
|
|
|
|
|
|
|
# And an outgoing activity was queued
|
|
|
|
outgoing_activity = db.query(models.OutgoingActivity).one()
|
|
|
|
assert outgoing_activity.outbox_object_id == outbox_object.id
|
|
|
|
assert outgoing_activity.recipient == ra.inbox_url
|
|
|
|
|
|
|
|
|
|
|
|
def test_send_create_activity__with_followers(
|
|
|
|
db: Session,
|
|
|
|
client: TestClient,
|
|
|
|
respx_mock: respx.MockRouter,
|
|
|
|
) -> None:
|
|
|
|
# given a remote actor
|
2022-07-26 20:26:34 +02:00
|
|
|
ra = setup_remote_actor(respx_mock)
|
2022-06-27 20:55:56 +02:00
|
|
|
|
2022-07-26 19:06:20 +02:00
|
|
|
# who is a follower
|
2022-07-26 20:26:34 +02:00
|
|
|
follower = setup_remote_actor_as_follower(ra)
|
2022-06-27 20:55:56 +02:00
|
|
|
|
|
|
|
with mock.patch.object(webfinger, "get_actor_url", return_value=ra.ap_id):
|
|
|
|
response = client.post(
|
|
|
|
"/admin/actions/new",
|
|
|
|
data={
|
|
|
|
"redirect_url": "http://testserver/",
|
|
|
|
"content": "hi followers",
|
|
|
|
"visibility": ap.VisibilityEnum.PUBLIC.name,
|
|
|
|
"csrf_token": generate_csrf_token(),
|
|
|
|
},
|
|
|
|
cookies=generate_admin_session_cookies(),
|
|
|
|
)
|
|
|
|
|
|
|
|
# Then the server returns a 302
|
|
|
|
assert response.status_code == 302
|
|
|
|
|
|
|
|
# And the Follow activity was created in the outbox
|
|
|
|
outbox_object = db.query(models.OutboxObject).one()
|
|
|
|
assert outbox_object.ap_type == "Note"
|
|
|
|
|
|
|
|
# And an outgoing activity was queued
|
|
|
|
outgoing_activity = db.query(models.OutgoingActivity).one()
|
|
|
|
assert outgoing_activity.outbox_object_id == outbox_object.id
|
|
|
|
assert outgoing_activity.recipient == follower.actor.inbox_url
|
2022-07-25 08:14:34 +02:00
|
|
|
|
|
|
|
|
|
|
|
def test_send_create_activity__question__one_of(
|
|
|
|
db: Session,
|
|
|
|
client: TestClient,
|
|
|
|
respx_mock: respx.MockRouter,
|
|
|
|
) -> None:
|
|
|
|
# given a remote actor
|
2022-07-26 20:26:34 +02:00
|
|
|
ra = setup_remote_actor(respx_mock)
|
2022-07-25 08:14:34 +02:00
|
|
|
|
2022-07-26 19:06:20 +02:00
|
|
|
# who is a follower
|
2022-07-26 20:26:34 +02:00
|
|
|
follower = setup_remote_actor_as_follower(ra)
|
2022-07-25 08:14:34 +02:00
|
|
|
|
|
|
|
with mock.patch.object(webfinger, "get_actor_url", return_value=ra.ap_id):
|
|
|
|
response = client.post(
|
|
|
|
"/admin/actions/new",
|
|
|
|
data={
|
|
|
|
"redirect_url": "http://testserver/",
|
|
|
|
"content": "hi followers",
|
|
|
|
"visibility": ap.VisibilityEnum.PUBLIC.name,
|
|
|
|
"csrf_token": generate_csrf_token(),
|
|
|
|
"poll_type": "oneOf",
|
|
|
|
"poll_duration": 5,
|
|
|
|
"poll_answer_1": "A",
|
|
|
|
"poll_answer_2": "B",
|
|
|
|
},
|
|
|
|
cookies=generate_admin_session_cookies(),
|
|
|
|
)
|
|
|
|
|
|
|
|
# Then the server returns a 302
|
|
|
|
assert response.status_code == 302
|
|
|
|
|
|
|
|
# And the Follow activity was created in the outbox
|
|
|
|
outbox_object = db.query(models.OutboxObject).one()
|
|
|
|
assert outbox_object.ap_type == "Question"
|
|
|
|
assert outbox_object.is_one_of_poll is True
|
|
|
|
assert len(outbox_object.poll_items) == 2
|
|
|
|
assert {pi["name"] for pi in outbox_object.poll_items} == {"A", "B"}
|
|
|
|
assert outbox_object.is_poll_ended is False
|
|
|
|
|
|
|
|
# And an outgoing activity was queued
|
|
|
|
outgoing_activity = db.query(models.OutgoingActivity).one()
|
|
|
|
assert outgoing_activity.outbox_object_id == outbox_object.id
|
|
|
|
assert outgoing_activity.recipient == follower.actor.inbox_url
|
|
|
|
|
|
|
|
|
|
|
|
def test_send_create_activity__question__any_of(
|
|
|
|
db: Session,
|
|
|
|
client: TestClient,
|
|
|
|
respx_mock: respx.MockRouter,
|
|
|
|
) -> None:
|
|
|
|
# given a remote actor
|
2022-07-26 20:26:34 +02:00
|
|
|
ra = setup_remote_actor(respx_mock)
|
2022-07-25 08:14:34 +02:00
|
|
|
|
2022-07-26 19:06:20 +02:00
|
|
|
# who is a follower
|
2022-07-26 20:26:34 +02:00
|
|
|
follower = setup_remote_actor_as_follower(ra)
|
2022-07-25 08:14:34 +02:00
|
|
|
|
|
|
|
with mock.patch.object(webfinger, "get_actor_url", return_value=ra.ap_id):
|
|
|
|
response = client.post(
|
|
|
|
"/admin/actions/new",
|
|
|
|
data={
|
|
|
|
"redirect_url": "http://testserver/",
|
|
|
|
"content": "hi followers",
|
|
|
|
"visibility": ap.VisibilityEnum.PUBLIC.name,
|
|
|
|
"csrf_token": generate_csrf_token(),
|
|
|
|
"poll_type": "anyOf",
|
|
|
|
"poll_duration": 10,
|
|
|
|
"poll_answer_1": "A",
|
|
|
|
"poll_answer_2": "B",
|
|
|
|
"poll_answer_3": "C",
|
|
|
|
"poll_answer_4": "D",
|
|
|
|
},
|
|
|
|
cookies=generate_admin_session_cookies(),
|
|
|
|
)
|
|
|
|
|
|
|
|
# Then the server returns a 302
|
|
|
|
assert response.status_code == 302
|
|
|
|
|
|
|
|
# And the Follow activity was created in the outbox
|
|
|
|
outbox_object = db.query(models.OutboxObject).one()
|
|
|
|
assert outbox_object.ap_type == "Question"
|
|
|
|
assert outbox_object.is_one_of_poll is False
|
|
|
|
assert len(outbox_object.poll_items) == 4
|
|
|
|
assert {pi["name"] for pi in outbox_object.poll_items} == {"A", "B", "C", "D"}
|
|
|
|
assert outbox_object.is_poll_ended is False
|
|
|
|
|
|
|
|
# And an outgoing activity was queued
|
|
|
|
outgoing_activity = db.query(models.OutgoingActivity).one()
|
|
|
|
assert outgoing_activity.outbox_object_id == outbox_object.id
|
|
|
|
assert outgoing_activity.recipient == follower.actor.inbox_url
|
2022-07-26 19:06:20 +02:00
|
|
|
|
|
|
|
|
|
|
|
def test_send_create_activity__article(
|
|
|
|
db: Session,
|
|
|
|
client: TestClient,
|
|
|
|
respx_mock: respx.MockRouter,
|
|
|
|
) -> None:
|
|
|
|
# given a remote actor
|
2022-07-26 20:26:34 +02:00
|
|
|
ra = setup_remote_actor(respx_mock)
|
2022-07-26 19:06:20 +02:00
|
|
|
|
|
|
|
# who is a follower
|
2022-07-26 20:26:34 +02:00
|
|
|
follower = setup_remote_actor_as_follower(ra)
|
2022-07-26 19:06:20 +02:00
|
|
|
|
|
|
|
with mock.patch.object(webfinger, "get_actor_url", return_value=ra.ap_id):
|
|
|
|
response = client.post(
|
|
|
|
"/admin/actions/new",
|
|
|
|
data={
|
|
|
|
"redirect_url": "http://testserver/",
|
|
|
|
"content": "hi followers",
|
|
|
|
"visibility": ap.VisibilityEnum.PUBLIC.name,
|
|
|
|
"csrf_token": generate_csrf_token(),
|
|
|
|
"name": "Article",
|
|
|
|
},
|
|
|
|
cookies=generate_admin_session_cookies(),
|
|
|
|
)
|
|
|
|
|
|
|
|
# Then the server returns a 302
|
|
|
|
assert response.status_code == 302
|
|
|
|
|
|
|
|
# And the Follow activity was created in the outbox
|
|
|
|
outbox_object = db.query(models.OutboxObject).one()
|
|
|
|
assert outbox_object.ap_type == "Article"
|
|
|
|
assert outbox_object.ap_object["name"] == "Article"
|
|
|
|
|
|
|
|
# And an outgoing activity was queued
|
|
|
|
outgoing_activity = db.query(models.OutgoingActivity).one()
|
|
|
|
assert outgoing_activity.outbox_object_id == outbox_object.id
|
|
|
|
assert outgoing_activity.recipient == follower.actor.inbox_url
|