2018-05-26 02:57:27 -05:00
|
|
|
import time
|
2018-05-26 02:50:59 -05:00
|
|
|
import os
|
|
|
|
|
|
|
|
import requests
|
|
|
|
from html2text import html2text
|
2018-05-27 05:02:14 -05:00
|
|
|
from utils import activitypub_utils
|
2018-05-26 02:50:59 -05:00
|
|
|
|
|
|
|
|
|
|
|
def resp2plaintext(resp):
|
|
|
|
"""Convert the body of a requests reponse to plain text in order to make basic assertions."""
|
|
|
|
return html2text(resp.text)
|
|
|
|
|
|
|
|
|
2018-05-26 03:43:05 -05:00
|
|
|
class Instance(object):
|
|
|
|
"""Test instance wrapper."""
|
2018-05-26 02:50:59 -05:00
|
|
|
|
2018-05-26 03:43:05 -05:00
|
|
|
def __init__(self, host_url, docker_url=None):
|
|
|
|
self.host_url = host_url
|
|
|
|
self.docker_url = docker_url or host_url
|
|
|
|
self.session = requests.Session()
|
2018-05-26 02:50:59 -05:00
|
|
|
|
2018-05-27 05:02:14 -05:00
|
|
|
def _do_req(self, url, headers):
|
|
|
|
url = url.replace(self.docker_url, self.host_url)
|
|
|
|
resp = requests.get(url, headers=headers)
|
|
|
|
resp.raise_for_status()
|
|
|
|
return resp.json()
|
|
|
|
|
|
|
|
def _parse_collection(self, payload=None, url=None):
|
|
|
|
return activitypub_utils.parse_collection(url=url, payload=payload, do_req=self._do_req)
|
|
|
|
|
2018-05-26 03:43:05 -05:00
|
|
|
def ping(self):
|
|
|
|
"""Ensures the homepage is reachable."""
|
|
|
|
resp = self.session.get(f'{self.host_url}/')
|
|
|
|
resp.raise_for_status()
|
|
|
|
assert resp.status_code == 200
|
2018-05-26 02:50:59 -05:00
|
|
|
|
2018-05-27 04:50:09 -05:00
|
|
|
def debug(self):
|
|
|
|
resp = self.session.get(f'{self.host_url}/api/debug', headers={'Accept': 'application/json'})
|
|
|
|
resp.raise_for_status()
|
|
|
|
|
|
|
|
return resp.json()
|
|
|
|
|
|
|
|
def drop_db(self):
|
|
|
|
resp = self.session.delete(f'{self.host_url}/api/debug', headers={'Accept': 'application/json'})
|
|
|
|
resp.raise_for_status()
|
|
|
|
|
|
|
|
return resp.json()
|
|
|
|
|
2018-05-26 03:43:05 -05:00
|
|
|
def login(self):
|
|
|
|
resp = self.session.post(f'{self.host_url}/login', data={'pass': 'hello'})
|
|
|
|
resp.raise_for_status()
|
|
|
|
assert resp.status_code == 200
|
2018-05-26 02:50:59 -05:00
|
|
|
|
2018-05-26 03:43:05 -05:00
|
|
|
def follow(self, instance: 'Instance') -> None:
|
|
|
|
# Instance1 follows instance2
|
|
|
|
resp = self.session.get(f'{self.host_url}/api/follow', params={'actor': instance.docker_url})
|
|
|
|
assert resp.status_code == 201
|
2018-05-26 02:50:59 -05:00
|
|
|
|
2018-05-26 03:43:05 -05:00
|
|
|
# We need to wait for the Follow/Accept dance
|
|
|
|
time.sleep(10)
|
2018-05-27 07:21:06 -05:00
|
|
|
return resp.headers.get('microblogpub-created-activity')
|
|
|
|
|
|
|
|
def undo(self, oid: str) -> None:
|
|
|
|
resp = self.session.get(f'{self.host_url}/api/undo', params={'id': oid})
|
|
|
|
assert resp.status_code == 201
|
|
|
|
|
|
|
|
# We need to wait for the Follow/Accept dance
|
|
|
|
time.sleep(10)
|
|
|
|
return resp.headers.get('microblogpub-created-activity')
|
2018-05-26 03:43:05 -05:00
|
|
|
|
|
|
|
def followers(self):
|
|
|
|
resp = self.session.get(f'{self.host_url}/followers', headers={'Accept': 'application/activity+json'})
|
|
|
|
resp.raise_for_status()
|
|
|
|
|
|
|
|
data = resp.json()
|
|
|
|
|
2018-05-27 05:02:14 -05:00
|
|
|
return self._parse_collection(payload=data)
|
2018-05-26 03:43:05 -05:00
|
|
|
|
|
|
|
def following(self):
|
|
|
|
resp = self.session.get(f'{self.host_url}/following', headers={'Accept': 'application/activity+json'})
|
|
|
|
resp.raise_for_status()
|
2018-05-26 02:50:59 -05:00
|
|
|
|
2018-05-26 03:43:05 -05:00
|
|
|
data = resp.json()
|
2018-05-26 02:50:59 -05:00
|
|
|
|
2018-05-27 05:02:14 -05:00
|
|
|
return self._parse_collection(payload=data)
|
2018-05-26 03:43:05 -05:00
|
|
|
|
2018-05-27 04:01:34 -05:00
|
|
|
def outbox(self):
|
|
|
|
resp = self.session.get(f'{self.host_url}/following', headers={'Accept': 'application/activity+json'})
|
|
|
|
resp.raise_for_status()
|
|
|
|
return resp.json()
|
|
|
|
|
2018-05-26 03:43:05 -05:00
|
|
|
|
2018-05-27 07:21:06 -05:00
|
|
|
def _instances():
|
2018-05-26 03:43:05 -05:00
|
|
|
instance1 = Instance('http://localhost:5006', 'http://instance1_web_1:5005')
|
|
|
|
instance1.ping()
|
|
|
|
|
|
|
|
instance2 = Instance('http://localhost:5007', 'http://instance2_web_1:5005')
|
|
|
|
instance2.ping()
|
|
|
|
|
|
|
|
# Login
|
|
|
|
instance1.login()
|
2018-05-27 04:52:27 -05:00
|
|
|
instance1.drop_db()
|
2018-05-26 03:43:05 -05:00
|
|
|
instance2.login()
|
2018-05-27 04:52:27 -05:00
|
|
|
instance2.drop_db()
|
2018-05-27 07:21:06 -05:00
|
|
|
|
|
|
|
return instance1, instance2
|
|
|
|
|
2018-05-26 03:43:05 -05:00
|
|
|
|
2018-05-27 07:21:06 -05:00
|
|
|
def test_follow():
|
|
|
|
instance1, instance2 = _instances()
|
2018-05-26 03:43:05 -05:00
|
|
|
# Instance1 follows instance2
|
|
|
|
instance1.follow(instance2)
|
2018-05-27 04:50:09 -05:00
|
|
|
instance1_debug = instance1.debug()
|
2018-05-27 07:21:06 -05:00
|
|
|
print(f'instance1_debug={instance1_debug}')
|
2018-05-27 04:50:09 -05:00
|
|
|
assert instance1_debug['inbox'] == 1 # An Accept activity should be there
|
|
|
|
assert instance1_debug['outbox'] == 1 # We've sent a Follow activity
|
|
|
|
|
|
|
|
instance2_debug = instance2.debug()
|
2018-05-27 07:21:06 -05:00
|
|
|
print(f'instance2_debug={instance2_debug}')
|
|
|
|
assert instance2_debug['inbox'] == 1 # An Follow activity should be there
|
|
|
|
assert instance2_debug['outbox'] == 1 # We've sent a Accept activity
|
2018-05-26 02:57:27 -05:00
|
|
|
|
2018-05-26 03:43:05 -05:00
|
|
|
assert instance2.followers() == [instance1.docker_url]
|
|
|
|
assert instance1.following() == [instance2.docker_url]
|
2018-05-27 07:21:06 -05:00
|
|
|
|
|
|
|
|
|
|
|
def test_follow_unfollow():
|
|
|
|
instance1, instance2 = _instances()
|
|
|
|
# Instance1 follows instance2
|
|
|
|
follow_id = instance1.follow(instance2)
|
|
|
|
instance1_debug = instance1.debug()
|
|
|
|
assert instance1_debug['inbox'] == 1 # An Accept activity should be there
|
|
|
|
assert instance1_debug['outbox'] == 1 # We've sent a Follow activity
|
|
|
|
|
|
|
|
instance2_debug = instance2.debug()
|
|
|
|
assert instance2_debug['inbox'] == 1 # An Follow activity should be there
|
|
|
|
assert instance2_debug['outbox'] == 1 # We've sent a Accept activity
|
|
|
|
|
|
|
|
assert instance2.followers() == [instance1.docker_url]
|
|
|
|
assert instance1.following() == [instance2.docker_url]
|
|
|
|
|
|
|
|
instance1.undo(follow_id)
|
|
|
|
|
|
|
|
assert instance2.followers() == []
|
|
|
|
assert instance1.following() == []
|
|
|
|
|
|
|
|
instance1_debug = instance1.debug()
|
|
|
|
assert instance1_debug['inbox'] == 1 # An Accept activity should be there
|
|
|
|
assert instance1_debug['outbox'] == 2 # We've sent a Follow and a Undo activity
|
|
|
|
|
|
|
|
instance2_debug = instance2.debug()
|
|
|
|
assert instance2_debug['inbox'] == 2 # An Follow and Undo activity should be there
|
|
|
|
assert instance2_debug['outbox'] == 1 # We've sent a Accept activity
|
|
|
|
|
|
|
|
|