2023-11-13 22:35:54 +00:00
|
|
|
"""Integration tests."""
|
|
|
|
from unittest.mock import patch
|
|
|
|
|
|
|
|
from arroba.datastore_storage import DatastoreStorage
|
|
|
|
from arroba.repo import Repo
|
|
|
|
from flask import g
|
|
|
|
from oauth_dropins.webutil.testutil import requests_response
|
|
|
|
|
|
|
|
from activitypub import ActivityPub
|
|
|
|
import app
|
|
|
|
from atproto import ATProto
|
|
|
|
import hub
|
|
|
|
from models import Target
|
2024-03-13 04:41:05 +00:00
|
|
|
from web import Web
|
2023-11-13 22:35:54 +00:00
|
|
|
|
|
|
|
from .testutil import ATPROTO_KEY, TestCase
|
|
|
|
from . import test_atproto
|
2023-11-15 20:40:59 +00:00
|
|
|
from . import test_web
|
2023-11-13 22:35:54 +00:00
|
|
|
|
|
|
|
DID_DOC = {
|
|
|
|
**test_atproto.DID_DOC,
|
|
|
|
'id': 'did:plc:alice',
|
2024-03-13 04:41:05 +00:00
|
|
|
'alsoKnownAs': ['at://alice.com'],
|
2023-11-13 22:35:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class IntegrationTests(TestCase):
|
|
|
|
|
Revert "cache outbound HTTP request responses, locally to each inbound request"
This reverts commit 30debfc8faf730190bd51a3aef49df6c6bfbd50a.
seemed promising, but broke in production. Saw a lot of `IncompleteRead`s on both GETs and POSTs. Rolled back for now.
```
('Connection broken: IncompleteRead(9172 bytes read, -4586 more expected)', IncompleteRead(9172 bytes read, -4586 more expected))
...
File "oauth_dropins/webutil/util.py", line 1673, in call
resp = getattr((session or requests), fn)(url, *args, **kwargs)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "requests_cache/session.py", line 102, in get
return self.request('GET', url, params=params, **kwargs)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "requests_cache/session.py", line 158, in request
return super().request(method, url, *args, headers=headers, **kwargs) # type: ignore
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "requests/sessions.py", line 589, in request
resp = self.send(prep, **send_kwargs)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "requests_cache/session.py", line 205, in send
response = self._send_and_cache(request, actions, cached_response, **kwargs)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "requests_cache/session.py", line 233, in _send_and_cache
self.cache.save_response(response, actions.cache_key, actions.expires)
File "requests_cache/backends/base.py", line 89, in save_response
cached_response = CachedResponse.from_response(response, expires=expires)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "requests_cache/models/response.py", line 102, in from_response
obj.raw = CachedHTTPResponse.from_response(response)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "requests_cache/models/raw_response.py", line 69, in from_response
_ = response.content # This property reads, decodes, and stores response content
^^^^^^^^^^^^^^^^
File "requests/models.py", line 899, in content
self._content = b"".join(self.iter_content(CONTENT_CHUNK_SIZE)) or b""
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "requests/models.py", line 818, in generate
raise ChunkedEncodingError(e)
```
2024-03-08 21:24:28 +00:00
|
|
|
@patch('requests.post')
|
2023-11-13 22:35:54 +00:00
|
|
|
@patch('requests.get')
|
2024-02-28 18:57:30 +00:00
|
|
|
@patch('common.ENABLED_BRIDGES', new=[('activitypub', 'atproto')])
|
2023-11-13 22:35:54 +00:00
|
|
|
def test_atproto_notify_reply_to_activitypub(self, mock_get, mock_post):
|
|
|
|
"""ATProto poll notifications, deliver reply to ActivityPub.
|
|
|
|
|
|
|
|
ActivityPub original post http://inst/post by bob
|
2024-03-13 04:41:05 +00:00
|
|
|
ATProto reply 123 by alice.com (did:plc:alice)
|
2023-11-13 22:35:54 +00:00
|
|
|
|
|
|
|
https://github.com/snarfed/bridgy-fed/issues/720
|
|
|
|
"""
|
|
|
|
# setup
|
|
|
|
self.store_object(id='did:plc:alice', raw=DID_DOC)
|
|
|
|
alice = self.make_user(id='did:plc:alice', cls=ATProto)
|
|
|
|
|
|
|
|
storage = DatastoreStorage()
|
|
|
|
Repo.create(storage, 'did:plc:bob', signing_key=ATPROTO_KEY)
|
2023-11-16 03:08:06 +00:00
|
|
|
bob = self.make_user(
|
|
|
|
id='http://inst/bob',
|
|
|
|
cls=ActivityPub,
|
|
|
|
copies=[Target(uri='did:plc:bob', protocol='atproto')],
|
|
|
|
obj_as2={
|
|
|
|
'id': 'http://inst/bob',
|
|
|
|
'inbox': 'http://inst/bob/inbox',
|
|
|
|
})
|
2023-11-13 22:35:54 +00:00
|
|
|
|
|
|
|
self.store_object(id='http://inst/post', source_protocol='activitypub',
|
|
|
|
our_as1={
|
|
|
|
'objectType': 'note',
|
|
|
|
'author': 'http://inst/bob',
|
|
|
|
},
|
|
|
|
copies=[
|
|
|
|
Target(uri='at://did:plc:bob/app.bsky.feed.post/123', protocol='atproto'),
|
|
|
|
])
|
|
|
|
|
|
|
|
# ATProto listNotifications => receive
|
|
|
|
mock_get.side_effect = [
|
|
|
|
requests_response({
|
|
|
|
'cursor': '...',
|
|
|
|
'notifications': [{
|
|
|
|
'uri': 'at://did:plc:alice/app.bsky.feed.post/456',
|
|
|
|
'cid': '...',
|
|
|
|
'author': {
|
|
|
|
'$type': 'app.bsky.actor.defs#profileView',
|
|
|
|
'did': 'did:plc:alice',
|
2024-03-13 04:41:05 +00:00
|
|
|
'handle': 'alice.com',
|
2023-11-13 22:35:54 +00:00
|
|
|
},
|
|
|
|
'reason': 'reply',
|
|
|
|
'record': {
|
|
|
|
'$type': 'app.bsky.feed.post',
|
|
|
|
'text': 'I hereby reply',
|
|
|
|
'reply': {
|
|
|
|
'root': {
|
|
|
|
'cid': '...',
|
|
|
|
'uri': 'at://did:plc:bob/app.bsky.feed.post/123',
|
|
|
|
},
|
|
|
|
'parent': {
|
|
|
|
'cid': '...',
|
|
|
|
'uri': 'at://did:plc:bob/app.bsky.feed.post/123',
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}],
|
|
|
|
}),
|
|
|
|
]
|
|
|
|
|
|
|
|
resp = self.post('/queue/atproto-poll-notifs', client=hub.app.test_client())
|
|
|
|
self.assertEqual(200, resp.status_code)
|
|
|
|
|
2023-11-20 05:52:53 +00:00
|
|
|
web_test = test_web.WebTest()
|
|
|
|
web_test.user = alice
|
|
|
|
web_test.assert_deliveries(mock_post, ['http://inst/bob/inbox'], data={
|
2023-11-13 22:35:54 +00:00
|
|
|
'@context': 'https://www.w3.org/ns/activitystreams',
|
|
|
|
'type': 'Create',
|
|
|
|
'id': 'https://atproto.brid.gy/convert/ap/at://did:plc:alice/app.bsky.feed.post/456#bridgy-fed-create',
|
|
|
|
'actor': 'https://atproto.brid.gy/ap/did:plc:alice',
|
|
|
|
'published': '2022-01-02T03:04:05+00:00',
|
|
|
|
'object': {
|
|
|
|
'type': 'Note',
|
|
|
|
'id': 'https://atproto.brid.gy/convert/ap/at://did:plc:alice/app.bsky.feed.post/456',
|
|
|
|
'url': 'http://localhost/r/https://bsky.app/profile/did:plc:alice/post/456',
|
|
|
|
'attributedTo': 'https://atproto.brid.gy/ap/did:plc:alice',
|
|
|
|
'content': 'I hereby reply',
|
|
|
|
'contentMap': {'en': 'I hereby reply'},
|
|
|
|
'inReplyTo': 'http://inst/post',
|
|
|
|
'tag': [{'type': 'Mention', 'href': 'http://inst/bob'}],
|
|
|
|
'to': ['https://www.w3.org/ns/activitystreams#Public'],
|
|
|
|
'cc': ['http://inst/bob'],
|
|
|
|
},
|
|
|
|
'to': ['https://www.w3.org/ns/activitystreams#Public'],
|
|
|
|
})
|
2024-03-13 04:41:05 +00:00
|
|
|
|
|
|
|
@patch('requests.post', return_value=requests_response(''))
|
|
|
|
@patch('requests.get')
|
|
|
|
def test_atproto_follow_to_web(self, mock_get, mock_post):
|
|
|
|
"""ATProto poll notifications, deliver follow to Web.
|
|
|
|
|
|
|
|
ATProto user alice.com (did:plc:alice)
|
|
|
|
ATProto follow at://did:plc:alice/app.bsky.graph.follow/123
|
|
|
|
Web user bob.com
|
|
|
|
"""
|
|
|
|
# setup
|
|
|
|
self.store_object(id='did:plc:alice', raw=DID_DOC)
|
|
|
|
alice = self.make_user(id='did:plc:alice', cls=ATProto)
|
|
|
|
|
|
|
|
storage = DatastoreStorage()
|
|
|
|
Repo.create(storage, 'did:plc:bob', signing_key=ATPROTO_KEY)
|
|
|
|
bob = self.make_user(id='bob.com', cls=Web,
|
|
|
|
copies=[Target(uri='did:plc:bob', protocol='atproto')])
|
|
|
|
|
|
|
|
# ATProto listNotifications => receive
|
|
|
|
mock_get.side_effect = [
|
|
|
|
# ATProto listNotifications
|
|
|
|
requests_response({
|
|
|
|
'cursor': '...',
|
|
|
|
'notifications': [{
|
|
|
|
'uri': 'at://did:plc:alice/app.bsky.graph.follow/123',
|
|
|
|
'cid': '...',
|
|
|
|
'author': {
|
|
|
|
'$type': 'app.bsky.actor.defs#profileView',
|
|
|
|
'did': 'did:plc:alice',
|
|
|
|
'handle': 'alice.com',
|
|
|
|
},
|
|
|
|
'reason': 'follow',
|
|
|
|
'record': {
|
|
|
|
'$type': 'app.bsky.graph.follow',
|
|
|
|
'subject': 'did:plc:bob',
|
|
|
|
'createdAt': '2022-01-02T03:04:05.000Z',
|
|
|
|
},
|
|
|
|
}],
|
|
|
|
}),
|
|
|
|
# webmention discovery
|
|
|
|
test_web.WEBMENTION_REL_LINK,
|
|
|
|
]
|
|
|
|
|
|
|
|
resp = self.post('/queue/atproto-poll-notifs', client=hub.app.test_client())
|
|
|
|
self.assertEqual(200, resp.status_code)
|
|
|
|
|
|
|
|
self.assert_req(mock_get, 'https://bob.com/')
|
|
|
|
self.assert_req(mock_post, 'https://bob.com/webmention', data={
|
|
|
|
'source': 'https://atproto.brid.gy/convert/web/at://did:plc:alice/app.bsky.graph.follow/123',
|
|
|
|
'target': 'https://bob.com/',
|
|
|
|
}, allow_redirects=False, headers={'Accept': '*/*'})
|