kopia lustrzana https://github.com/snarfed/bridgy-fed
				
				
				
			
		
			
				
	
	
		
			482 wiersze
		
	
	
		
			18 KiB
		
	
	
	
		
			Python
		
	
	
			
		
		
	
	
			482 wiersze
		
	
	
		
			18 KiB
		
	
	
	
		
			Python
		
	
	
"""Unit tests for follow.py.
 | 
						|
"""
 | 
						|
import copy
 | 
						|
from unittest.mock import patch
 | 
						|
 | 
						|
from flask import get_flashed_messages, session
 | 
						|
from granary import as2
 | 
						|
from oauth_dropins import indieauth
 | 
						|
from oauth_dropins.webutil import util
 | 
						|
from oauth_dropins.webutil.testutil import requests_response
 | 
						|
from oauth_dropins.webutil.util import json_dumps, json_loads
 | 
						|
 | 
						|
# import first so that Fake is defined before URL routes are registered
 | 
						|
from .testutil import Fake, TestCase
 | 
						|
 | 
						|
from activitypub import ActivityPub
 | 
						|
import common
 | 
						|
from common import redirect_unwrap
 | 
						|
from models import Follower, Object, User
 | 
						|
from web import Web
 | 
						|
 | 
						|
WEBFINGER = requests_response({
 | 
						|
    'subject': 'acct:foo@bar',
 | 
						|
    'aliases': [
 | 
						|
        'https://bar/foo',
 | 
						|
    ],
 | 
						|
    'links': [{
 | 
						|
        'rel': 'http://ostatus.org/schema/1.0/subscribe',
 | 
						|
        'template': 'https://bar/follow?uri={uri}'
 | 
						|
    }, {
 | 
						|
        'rel': 'self',
 | 
						|
        'type': as2.CONTENT_TYPE,
 | 
						|
        'href': 'https://bar/actor'
 | 
						|
    }],
 | 
						|
})
 | 
						|
FOLLOWEE = {
 | 
						|
    'type': 'Person',
 | 
						|
    'id': 'https://bar/id',
 | 
						|
    'url': 'https://bar/url',
 | 
						|
    'inbox': 'http://bar/inbox',
 | 
						|
}
 | 
						|
FOLLOW_ADDRESS = {
 | 
						|
    '@context': 'https://www.w3.org/ns/activitystreams',
 | 
						|
    'type': 'Follow',
 | 
						|
    'id': f'http://localhost/web/alice.com/following#2022-01-02T03:04:05-@foo@bar',
 | 
						|
    'actor': 'http://localhost/alice.com',
 | 
						|
    'object': FOLLOWEE,
 | 
						|
    'to': [as2.PUBLIC_AUDIENCE],
 | 
						|
}
 | 
						|
FOLLOW_URL = copy.deepcopy(FOLLOW_ADDRESS)
 | 
						|
FOLLOW_URL['id'] = f'http://localhost/web/alice.com/following#2022-01-02T03:04:05-https://bar/actor'
 | 
						|
UNDO_FOLLOW = {
 | 
						|
    '@context': 'https://www.w3.org/ns/activitystreams',
 | 
						|
    'type': 'Undo',
 | 
						|
    'id': f'http://localhost/web/alice.com/following#undo-2022-01-02T03:04:05-https://bar/id',
 | 
						|
    'actor': 'http://localhost/alice.com',
 | 
						|
    'object': FOLLOW_ADDRESS,
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
@patch('requests.get')
 | 
						|
class RemoteFollowTest(TestCase):
 | 
						|
 | 
						|
    def setUp(self):
 | 
						|
        super().setUp()
 | 
						|
        self.make_user('user.com')
 | 
						|
 | 
						|
    def test_no_domain(self, _):
 | 
						|
        got = self.client.post('/remote-follow?address=@foo@bar&protocol=web')
 | 
						|
        self.assertEqual(400, got.status_code)
 | 
						|
 | 
						|
    def test_no_address(self, _):
 | 
						|
        got = self.client.post('/remote-follow?domain=baz.com&protocol=web')
 | 
						|
        self.assertEqual(400, got.status_code)
 | 
						|
 | 
						|
    def test_no_protocol(self, _):
 | 
						|
        got = self.client.post('/remote-follow?address=@foo@bar&domain=user.com')
 | 
						|
        self.assertEqual(400, got.status_code)
 | 
						|
 | 
						|
    def test_unknown_protocol(self, _):
 | 
						|
        got = self.client.post('/remote-follow?address=@foo@bar&domain=user.com&protocol=foo')
 | 
						|
        self.assertEqual(400, got.status_code)
 | 
						|
 | 
						|
    def test_no_user(self, _):
 | 
						|
        got = self.client.post('/remote-follow?address=@foo@bar&domain=baz.com')
 | 
						|
        self.assertEqual(400, got.status_code)
 | 
						|
 | 
						|
    def test(self, mock_get):
 | 
						|
        mock_get.return_value = WEBFINGER
 | 
						|
        got = self.client.post('/remote-follow?address=@foo@bar&domain=user.com&protocol=web')
 | 
						|
        self.assertEqual(302, got.status_code)
 | 
						|
        self.assertEqual('https://bar/follow?uri=@user.com@user.com',
 | 
						|
                         got.headers['Location'])
 | 
						|
 | 
						|
        mock_get.assert_has_calls((
 | 
						|
            self.req('https://bar/.well-known/webfinger?resource=acct:foo@bar'),
 | 
						|
        ))
 | 
						|
 | 
						|
    def test_url(self, mock_get):
 | 
						|
        mock_get.return_value = WEBFINGER
 | 
						|
        got = self.client.post('/remote-follow?address=https://bar/foo&domain=user.com&protocol=web')
 | 
						|
        self.assertEqual(302, got.status_code)
 | 
						|
        self.assertEqual('https://bar/follow?uri=@user.com@user.com', got.headers['Location'])
 | 
						|
 | 
						|
        mock_get.assert_has_calls((
 | 
						|
            self.req('https://bar/.well-known/webfinger?resource=https://bar/foo'),
 | 
						|
        ))
 | 
						|
 | 
						|
    def test_no_webfinger_subscribe_link(self, mock_get):
 | 
						|
        mock_get.return_value = requests_response({
 | 
						|
            'subject': 'acct:foo@bar',
 | 
						|
            'links': [{'rel': 'other', 'template': 'meh'}],
 | 
						|
        })
 | 
						|
 | 
						|
        got = self.client.post('/remote-follow?address=https://bar/foo&domain=user.com&protocol=web')
 | 
						|
        self.assertEqual(302, got.status_code)
 | 
						|
        self.assertEqual('/web/user.com', got.headers['Location'])
 | 
						|
 | 
						|
    def test_webfinger_error(self, mock_get):
 | 
						|
        mock_get.return_value = requests_response(status=500)
 | 
						|
 | 
						|
        got = self.client.post('/remote-follow?address=https://bar/foo&domain=user.com&protocol=web')
 | 
						|
        self.assertEqual(302, got.status_code)
 | 
						|
        self.assertEqual('/web/user.com', got.headers['Location'])
 | 
						|
 | 
						|
    def test_webfinger_returns_not_json(self, mock_get):
 | 
						|
        mock_get.return_value = requests_response('<html>not json</html>')
 | 
						|
 | 
						|
        got = self.client.post('/remote-follow?address=https://bar/foo&domain=user.com&protocol=web')
 | 
						|
        self.assertEqual(302, got.status_code)
 | 
						|
        self.assertEqual('/web/user.com', got.headers['Location'])
 | 
						|
 | 
						|
 | 
						|
@patch('requests.post')
 | 
						|
@patch('requests.get')
 | 
						|
class FollowTest(TestCase):
 | 
						|
 | 
						|
    def setUp(self):
 | 
						|
        super().setUp()
 | 
						|
        self.user = self.make_user('alice.com')
 | 
						|
        self.state = {
 | 
						|
            'endpoint': 'http://auth/endpoint',
 | 
						|
            'me': 'https://alice.com',
 | 
						|
            'state': '@foo@bar',
 | 
						|
        }
 | 
						|
 | 
						|
    def test_start(self, mock_get, _):
 | 
						|
        mock_get.return_value = requests_response('')  # IndieAuth endpoint discovery
 | 
						|
 | 
						|
        resp = self.client.post('/follow/start', data={
 | 
						|
            'me': 'https://alice.com',
 | 
						|
            'address': '@foo@bar',
 | 
						|
        })
 | 
						|
        self.assertEqual(302, resp.status_code)
 | 
						|
        self.assertTrue(resp.headers['Location'].startswith(indieauth.INDIEAUTH_URL),
 | 
						|
                        resp.headers['Location'])
 | 
						|
 | 
						|
    def test_callback_address(self, mock_get, mock_post):
 | 
						|
        mock_get.side_effect = (
 | 
						|
            # oauth-dropins indieauth https://alice.com fetch for user json
 | 
						|
            requests_response(''),
 | 
						|
            WEBFINGER,
 | 
						|
            self.as2_resp(FOLLOWEE),
 | 
						|
        )
 | 
						|
        mock_post.side_effect = (
 | 
						|
            requests_response('me=https://alice.com'),
 | 
						|
            requests_response('OK'),  # AP Follow to inbox
 | 
						|
        )
 | 
						|
 | 
						|
        state = util.encode_oauth_state(self.state)
 | 
						|
        resp = self.client.get(f'/follow/callback?code=my_code&state={state}')
 | 
						|
        self.check('@foo@bar', resp, FOLLOW_ADDRESS, mock_get, mock_post)
 | 
						|
        mock_get.assert_has_calls((
 | 
						|
            self.req('https://bar/.well-known/webfinger?resource=acct:foo@bar'),
 | 
						|
        ))
 | 
						|
 | 
						|
    def test_callback_url(self, mock_get, mock_post):
 | 
						|
        mock_get.side_effect = (
 | 
						|
            requests_response(''),
 | 
						|
            self.as2_resp(FOLLOWEE),
 | 
						|
        )
 | 
						|
        mock_post.side_effect = (
 | 
						|
            requests_response('me=https://alice.com'),
 | 
						|
            requests_response('OK'),  # AP Follow to inbox
 | 
						|
        )
 | 
						|
 | 
						|
        self.state['state'] = 'https://bar/actor'
 | 
						|
        state = util.encode_oauth_state(self.state)
 | 
						|
        resp = self.client.get(f'/follow/callback?code=my_code&state={state}')
 | 
						|
        self.check('https://bar/actor', resp, FOLLOW_URL, mock_get, mock_post)
 | 
						|
 | 
						|
    def check(self, input, resp, expected_follow, mock_get, mock_post):
 | 
						|
        self.assertEqual(302, resp.status_code)
 | 
						|
        self.assertEqual('/web/alice.com/following',resp.headers['Location'])
 | 
						|
        self.assertEqual([f'Followed <a href="https://bar/url">{input}</a>.'],
 | 
						|
                         get_flashed_messages())
 | 
						|
 | 
						|
        mock_get.assert_has_calls((
 | 
						|
            self.as2_req('https://bar/actor'),
 | 
						|
        ))
 | 
						|
        inbox_args, inbox_kwargs = mock_post.call_args
 | 
						|
        self.assertEqual(('http://bar/inbox',), inbox_args)
 | 
						|
        self.assert_equals(expected_follow, json_loads(inbox_kwargs['data']))
 | 
						|
 | 
						|
        # check that we signed with the follower's key
 | 
						|
        sig_template = inbox_kwargs['auth'].header_signer.signature_template
 | 
						|
        self.assertTrue(sig_template.startswith('keyId="http://localhost/alice.com"'),
 | 
						|
                        sig_template)
 | 
						|
 | 
						|
        follow_id = f'http://localhost/web/alice.com/following#2022-01-02T03:04:05-{input}'
 | 
						|
 | 
						|
        followers = Follower.query().fetch()
 | 
						|
        followee = ActivityPub(id='https://bar/id').key
 | 
						|
        self.assert_entities_equal(
 | 
						|
            Follower(from_=self.user.key, to=followee,
 | 
						|
                     follow=Object(id=follow_id).key, status='active'),
 | 
						|
            followers,
 | 
						|
            ignore=['created', 'updated'])
 | 
						|
 | 
						|
        self.assert_object(follow_id, users=[self.user.key, followee],
 | 
						|
                           status='complete', labels=['user', 'activity'],
 | 
						|
                           source_protocol='ui', as2=expected_follow,
 | 
						|
                           as1=as2.to_as1(expected_follow))
 | 
						|
 | 
						|
        self.assertEqual('https://alice.com', session['indieauthed-me'])
 | 
						|
 | 
						|
    def test_callback_missing_user(self, mock_get, mock_post):
 | 
						|
        self.user.key.delete()
 | 
						|
        mock_post.return_value = requests_response('me=https://alice.com')
 | 
						|
        state = util.encode_oauth_state(self.state)
 | 
						|
        resp = self.client.get(f'/follow/callback?code=my_code&state={state}')
 | 
						|
        self.assertEqual(400, resp.status_code)
 | 
						|
 | 
						|
    def test_callback_user_use_instead(self, mock_get, mock_post):
 | 
						|
        user = self.make_user('www.alice.com')
 | 
						|
        self.user.use_instead = user.key
 | 
						|
        self.user.put()
 | 
						|
 | 
						|
        mock_get.side_effect = (
 | 
						|
            requests_response(''),
 | 
						|
            self.as2_resp(FOLLOWEE),
 | 
						|
        )
 | 
						|
        mock_post.side_effect = (
 | 
						|
            requests_response('me=https://alice.com'),
 | 
						|
            requests_response('OK'),  # AP Follow to inbox
 | 
						|
        )
 | 
						|
 | 
						|
        self.state['state'] = 'https://bar/actor'
 | 
						|
        state = util.encode_oauth_state(self.state)
 | 
						|
        resp = self.client.get(f'/follow/callback?code=my_code&state={state}')
 | 
						|
        self.assertEqual(302, resp.status_code)
 | 
						|
        self.assertEqual('/web/www.alice.com/following', resp.headers['Location'])
 | 
						|
 | 
						|
        id = 'http://localhost/web/www.alice.com/following#2022-01-02T03:04:05-https://bar/actor'
 | 
						|
        expected_follow = {
 | 
						|
            '@context': 'https://www.w3.org/ns/activitystreams',
 | 
						|
            'type': 'Follow',
 | 
						|
            'id': id,
 | 
						|
            'actor': 'http://localhost/www.alice.com',
 | 
						|
            'object': FOLLOWEE,
 | 
						|
            'to': [as2.PUBLIC_AUDIENCE],
 | 
						|
        }
 | 
						|
        followee = ActivityPub(id='https://bar/id').key
 | 
						|
        follow_obj = self.assert_object(
 | 
						|
            id, users=[user.key, followee], status='complete',
 | 
						|
            labels=['user', 'activity'], source_protocol='ui', as2=expected_follow,
 | 
						|
            as1=as2.to_as1(expected_follow))
 | 
						|
 | 
						|
        followers = Follower.query().fetch()
 | 
						|
        self.assert_entities_equal(
 | 
						|
            Follower(from_=user.key, to=followee, follow=follow_obj.key, status='active'),
 | 
						|
            followers,
 | 
						|
            ignore=['created', 'updated'])
 | 
						|
 | 
						|
    def test_indieauthed_session(self, mock_get, mock_post):
 | 
						|
        mock_get.side_effect = (
 | 
						|
            self.as2_resp(FOLLOWEE),
 | 
						|
        )
 | 
						|
        mock_post.side_effect = (
 | 
						|
            requests_response('OK'),  # AP Follow to inbox
 | 
						|
        )
 | 
						|
 | 
						|
        with self.client.session_transaction() as ctx_session:
 | 
						|
            ctx_session['indieauthed-me'] = 'https://alice.com'
 | 
						|
 | 
						|
        resp = self.client.post('/follow/start', data={
 | 
						|
            'me': 'https://alice.com',
 | 
						|
            'address': 'https://bar/actor',
 | 
						|
        })
 | 
						|
        self.check('https://bar/actor', resp, FOLLOW_URL, mock_get, mock_post)
 | 
						|
 | 
						|
    def test_indieauthed_session_wrong_me(self, mock_get, mock_post):
 | 
						|
        mock_get.side_effect = (
 | 
						|
            requests_response(''),  # IndieAuth endpoint discovery
 | 
						|
        )
 | 
						|
 | 
						|
        with self.client.session_transaction() as ctx_session:
 | 
						|
            ctx_session['indieauthed-me'] = 'https://eve.com'
 | 
						|
 | 
						|
        resp = self.client.post('/follow/start', data={
 | 
						|
            'me': 'https://alice.com',
 | 
						|
            'address': 'https://bar/actor',
 | 
						|
        })
 | 
						|
        self.assertEqual(302, resp.status_code)
 | 
						|
        self.assertTrue(resp.headers['Location'].startswith(indieauth.INDIEAUTH_URL),
 | 
						|
                        resp.headers['Location'])
 | 
						|
 | 
						|
 | 
						|
@patch('requests.post')
 | 
						|
@patch('requests.get')
 | 
						|
class UnfollowTest(TestCase):
 | 
						|
 | 
						|
    def setUp(self):
 | 
						|
        super().setUp()
 | 
						|
        self.user = self.make_user('alice.com')
 | 
						|
        self.follower = Follower.get_or_create(
 | 
						|
            from_=self.user,
 | 
						|
            to=ActivityPub.get_or_create('https://bar/id', actor_as2=FOLLOWEE),
 | 
						|
            follow=Object(id=FOLLOW_ADDRESS['id'], as2=FOLLOW_ADDRESS).put(),
 | 
						|
            status='active',
 | 
						|
        )
 | 
						|
 | 
						|
        self.state = util.encode_oauth_state({
 | 
						|
            'endpoint': 'http://auth/endpoint',
 | 
						|
            'me': 'https://alice.com',
 | 
						|
            'state': self.follower.key.id(),
 | 
						|
        })
 | 
						|
 | 
						|
    def test_start(self, mock_get, _):
 | 
						|
        mock_get.return_value = requests_response('')  # IndieAuth endpoint discovery
 | 
						|
 | 
						|
        resp = self.client.post('/unfollow/start', data={
 | 
						|
            'me': 'https://alice.com',
 | 
						|
            'key': self.follower.key.id(),
 | 
						|
        })
 | 
						|
        self.assertEqual(302, resp.status_code)
 | 
						|
        self.assertTrue(resp.headers['Location'].startswith(indieauth.INDIEAUTH_URL),
 | 
						|
                        resp.headers['Location'])
 | 
						|
 | 
						|
    def test_callback(self, mock_get, mock_post):
 | 
						|
        # oauth-dropins indieauth https://alice.com fetch for user json
 | 
						|
        mock_get.return_value = requests_response('')
 | 
						|
        mock_post.side_effect = (
 | 
						|
            requests_response('me=https://alice.com'),
 | 
						|
            requests_response('OK'),  # AP Undo Follow to inbox
 | 
						|
        )
 | 
						|
 | 
						|
        resp = self.client.get(f'/unfollow/callback?code=my_code&state={self.state}')
 | 
						|
        self.check(resp, UNDO_FOLLOW, mock_get, mock_post)
 | 
						|
 | 
						|
    def test_callback_last_follow_object_str(self, mock_get, mock_post):
 | 
						|
        to = self.follower.to.get()
 | 
						|
        to.actor_as2 = None
 | 
						|
        to.put()
 | 
						|
 | 
						|
        obj = self.follower.follow.get()
 | 
						|
        obj.as2['object'] = FOLLOWEE['id']
 | 
						|
        obj.put()
 | 
						|
 | 
						|
        mock_get.side_effect = (
 | 
						|
            # oauth-dropins indieauth https://alice.com fetch for user json
 | 
						|
            requests_response(''),
 | 
						|
            # actor fetch to discover inbox
 | 
						|
            self.as2_resp(FOLLOWEE),
 | 
						|
        )
 | 
						|
        mock_post.side_effect = (
 | 
						|
            requests_response('me=https://alice.com'),
 | 
						|
            requests_response('OK'),  # AP Undo Follow to inbox
 | 
						|
        )
 | 
						|
 | 
						|
        undo = copy.deepcopy(UNDO_FOLLOW)
 | 
						|
        undo['object']['object'] = FOLLOWEE['id']
 | 
						|
 | 
						|
        resp = self.client.get(f'/unfollow/callback?code=my_code&state={self.state}')
 | 
						|
        self.check(resp, undo, mock_get, mock_post)
 | 
						|
 | 
						|
    def check(self, resp, expected_undo, mock_get, mock_post):
 | 
						|
        self.assertEqual(302, resp.status_code)
 | 
						|
        self.assertEqual('/web/alice.com/following', resp.headers['Location'])
 | 
						|
        self.assertEqual([f'Unfollowed <a href="https://bar/url">bar/url</a>.'],
 | 
						|
                         get_flashed_messages())
 | 
						|
 | 
						|
        inbox_args, inbox_kwargs = mock_post.call_args
 | 
						|
        self.assertEqual(('http://bar/inbox',), inbox_args)
 | 
						|
        self.assert_equals(expected_undo, json_loads(inbox_kwargs['data']))
 | 
						|
 | 
						|
        # check that we signed with the follower's key
 | 
						|
        sig_template = inbox_kwargs['auth'].header_signer.signature_template
 | 
						|
        self.assertTrue(sig_template.startswith('keyId="http://localhost/alice.com"'),
 | 
						|
                        sig_template)
 | 
						|
 | 
						|
        follower = Follower.query().get()
 | 
						|
        self.assertEqual('inactive', follower.status)
 | 
						|
 | 
						|
        self.assert_object(
 | 
						|
            'http://localhost/web/alice.com/following#undo-2022-01-02T03:04:05-https://bar/id',
 | 
						|
            users=[self.user.key], status='complete', source_protocol='ui',
 | 
						|
            labels=['user', 'activity'], as2=expected_undo, as1=as2.to_as1(expected_undo))
 | 
						|
 | 
						|
        self.assertEqual('https://alice.com', session['indieauthed-me'])
 | 
						|
 | 
						|
    def test_callback_user_use_instead(self, mock_get, mock_post):
 | 
						|
        user = self.make_user('www.alice.com')
 | 
						|
        self.user.use_instead = user.key
 | 
						|
        self.user.put()
 | 
						|
 | 
						|
        Follower.get_or_create(
 | 
						|
            from_=self.user,
 | 
						|
            to=ActivityPub.get_or_create('https://bar/id', actor_as2=FOLLOWEE),
 | 
						|
            follow=Object(id=FOLLOW_ADDRESS['id'], as2=FOLLOW_ADDRESS).put(),
 | 
						|
            status='active')
 | 
						|
 | 
						|
        mock_get.side_effect = (
 | 
						|
            requests_response(''),
 | 
						|
            self.as2_resp(FOLLOWEE),
 | 
						|
        )
 | 
						|
        mock_post.side_effect = (
 | 
						|
            requests_response('me=https://alice.com'),
 | 
						|
            requests_response('OK'),  # AP Undo Follow to inbox
 | 
						|
        )
 | 
						|
 | 
						|
        state = util.encode_oauth_state({
 | 
						|
            'endpoint': 'http://auth/endpoint',
 | 
						|
            'me': 'https://alice.com',
 | 
						|
            'state': self.follower.key.id(),
 | 
						|
        })
 | 
						|
        resp = self.client.get(f'/unfollow/callback?code=my_code&state={state}')
 | 
						|
        self.assertEqual(302, resp.status_code)
 | 
						|
        self.assertEqual('/web/www.alice.com/following', resp.headers['Location'])
 | 
						|
 | 
						|
        id = 'http://localhost/web/www.alice.com/following#undo-2022-01-02T03:04:05-https://bar/id'
 | 
						|
        expected_undo = {
 | 
						|
            '@context': 'https://www.w3.org/ns/activitystreams',
 | 
						|
            'type': 'Undo',
 | 
						|
            'id': id,
 | 
						|
            'actor': 'http://localhost/www.alice.com',
 | 
						|
            'object': FOLLOW_ADDRESS,
 | 
						|
        }
 | 
						|
 | 
						|
        inbox_args, inbox_kwargs = mock_post.call_args_list[1]
 | 
						|
        self.assertEqual(('http://bar/inbox',), inbox_args)
 | 
						|
        self.assert_equals(expected_undo, json_loads(inbox_kwargs['data']))
 | 
						|
 | 
						|
        follower = Follower.query().get()
 | 
						|
        self.assertEqual('inactive', follower.status)
 | 
						|
 | 
						|
        self.assert_object(id, users=[user.key], status='complete',
 | 
						|
                           source_protocol='ui', labels=['user', 'activity'],
 | 
						|
                           as2=expected_undo, as1=as2.to_as1(expected_undo))
 | 
						|
 | 
						|
    def test_indieauthed_session(self, mock_get, mock_post):
 | 
						|
        # oauth-dropins indieauth https://alice.com fetch for user json
 | 
						|
        mock_get.return_value = requests_response('')
 | 
						|
        mock_post.side_effect = (
 | 
						|
            requests_response('OK'),  # AP Undo Follow to inbox
 | 
						|
        )
 | 
						|
 | 
						|
        with self.client.session_transaction() as ctx_session:
 | 
						|
            ctx_session['indieauthed-me'] = 'https://alice.com'
 | 
						|
 | 
						|
        resp = self.client.post('/unfollow/start', data={
 | 
						|
            'me': 'https://alice.com',
 | 
						|
            'key': self.follower.key.id(),
 | 
						|
        })
 | 
						|
        self.check(resp, UNDO_FOLLOW, mock_get, mock_post)
 | 
						|
 | 
						|
    def test_indieauthed_session_wrong_me(self, mock_get, mock_post):
 | 
						|
        mock_get.side_effect = (
 | 
						|
            requests_response(''),  # IndieAuth endpoint discovery
 | 
						|
        )
 | 
						|
 | 
						|
        with self.client.session_transaction() as ctx_session:
 | 
						|
            ctx_session['indieauthed-me'] = 'https://eve.com'
 | 
						|
 | 
						|
        resp = self.client.post('/unfollow/start', data={
 | 
						|
            'me': 'https://alice.com',
 | 
						|
            'key': self.follower.key.id(),
 | 
						|
        })
 | 
						|
        self.assertEqual(302, resp.status_code)
 | 
						|
        self.assertTrue(resp.headers['Location'].startswith(indieauth.INDIEAUTH_URL),
 | 
						|
                        resp.headers['Location'])
 |