2017-07-26 18:32:35 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
try:
|
|
|
|
from unittest.mock import patch
|
|
|
|
except ImportError:
|
|
|
|
from mock import patch
|
|
|
|
|
2017-07-28 15:31:37 +02:00
|
|
|
import django
|
2017-07-26 18:32:35 +02:00
|
|
|
from django.conf import settings
|
|
|
|
from django.test import TestCase
|
|
|
|
|
|
|
|
import cas
|
|
|
|
|
|
|
|
from allauth_cas import CAS_PROVIDER_SESSION_KEY
|
|
|
|
|
2017-07-28 15:31:37 +02:00
|
|
|
if django.VERSION >= (1, 10):
|
|
|
|
from django.urls import reverse
|
|
|
|
else:
|
|
|
|
from django.core.urlresolvers import reverse
|
|
|
|
|
2017-07-26 18:32:35 +02:00
|
|
|
|
|
|
|
class CASTestCase(TestCase):
|
|
|
|
|
2017-07-28 15:31:37 +02:00
|
|
|
def client_cas_login(
|
|
|
|
self,
|
|
|
|
client, provider_id='theid',
|
|
|
|
username=None, attributes={}):
|
|
|
|
"""
|
|
|
|
Authenticate client through provider_id.
|
|
|
|
|
|
|
|
Returns the response of the callback view.
|
|
|
|
|
|
|
|
username and attributes control the CAS server response when ticket is
|
|
|
|
checked.
|
|
|
|
"""
|
2017-12-29 22:21:44 +01:00
|
|
|
client.get(reverse('{id}_login'.format(id=provider_id)))
|
2017-07-28 15:31:37 +02:00
|
|
|
self.patch_cas_response(
|
|
|
|
valid_ticket='__all__',
|
|
|
|
username=username, attributes=attributes,
|
|
|
|
)
|
|
|
|
callback_url = reverse('{id}_callback'.format(id=provider_id))
|
|
|
|
r = client.get(callback_url, {'ticket': 'fake-ticket'})
|
|
|
|
self.patch_cas_response_stop()
|
|
|
|
return r
|
|
|
|
|
2017-07-26 18:32:35 +02:00
|
|
|
def patch_cas_response(
|
|
|
|
self,
|
|
|
|
valid_ticket,
|
|
|
|
username=None, attributes={}):
|
|
|
|
"""
|
|
|
|
Patch the CASClient class used by views of CAS providers.
|
|
|
|
|
|
|
|
Arguments determines the response of verify_ticket method:
|
|
|
|
|
|
|
|
- If ticket given as paramater to this method is equal to valid_ticket,
|
|
|
|
its return value corresponds to a successful authentication on CAS
|
|
|
|
server for user whose login is username argument (default:
|
|
|
|
"username") and extra attributes (provided by the server) are
|
|
|
|
attributes argument (default: {}).
|
|
|
|
|
|
|
|
- If ticket doesn't match valid_ticket, the response corresponds to a
|
|
|
|
reject from CAS server.
|
|
|
|
|
|
|
|
Special values for valid_ticket:
|
|
|
|
|
|
|
|
- If valid_ticket is '__all__', a success response is always returned.
|
|
|
|
- If valid_ticket is None, a failure response is always returned.
|
|
|
|
|
|
|
|
Note that valid_ticket sould be a string (which is the type of the
|
|
|
|
ticket retrieved from GET parameter on request on the callback view).
|
|
|
|
"""
|
|
|
|
if hasattr(self, '_patch_cas_client'):
|
2017-07-28 15:31:37 +02:00
|
|
|
self.patch_cas_response_stop()
|
2017-07-26 18:32:35 +02:00
|
|
|
|
2017-07-28 15:31:37 +02:00
|
|
|
class MockCASClient(object):
|
2017-07-26 18:32:35 +02:00
|
|
|
_username = username
|
|
|
|
|
2017-07-28 15:31:37 +02:00
|
|
|
def __new__(self_client, *args, **kwargs):
|
|
|
|
version = kwargs.pop('version')
|
|
|
|
if version in (1, '1'):
|
|
|
|
client_class = cas.CASClientV1
|
|
|
|
elif version in (2, '2'):
|
|
|
|
client_class = cas.CASClientV2
|
|
|
|
elif version in (3, '3'):
|
|
|
|
client_class = cas.CASClientV3
|
|
|
|
elif version == 'CAS_2_SAML_1_0':
|
|
|
|
client_class = cas.CASClientWithSAMLV1
|
|
|
|
else:
|
|
|
|
raise ValueError('Unsupported CAS_VERSION %r' % version)
|
|
|
|
|
|
|
|
client_class._username = self_client._username
|
|
|
|
|
|
|
|
def verify_ticket(self, ticket):
|
|
|
|
if valid_ticket == '__all__' or ticket == valid_ticket:
|
|
|
|
username = self._username or 'username'
|
|
|
|
return username, attributes, None
|
|
|
|
return None, {}, None
|
|
|
|
|
|
|
|
patcher = patch.object(
|
|
|
|
client_class, 'verify_ticket',
|
|
|
|
new=verify_ticket,
|
|
|
|
)
|
|
|
|
patcher.start()
|
|
|
|
|
|
|
|
return client_class(*args, **kwargs)
|
2017-07-26 18:32:35 +02:00
|
|
|
|
|
|
|
self._patch_cas_client = patch(
|
|
|
|
'allauth_cas.views.cas.CASClient',
|
|
|
|
MockCASClient,
|
|
|
|
)
|
|
|
|
self._patch_cas_client.start()
|
|
|
|
|
2017-07-28 15:31:37 +02:00
|
|
|
def patch_cas_response_stop(self):
|
2017-07-26 18:32:35 +02:00
|
|
|
self._patch_cas_client.stop()
|
|
|
|
del self._patch_cas_client
|
|
|
|
|
|
|
|
def tearDown(self):
|
|
|
|
if hasattr(self, '_patch_cas_client'):
|
2017-07-28 15:31:37 +02:00
|
|
|
self.patch_cas_response_stop()
|
2017-07-26 18:32:35 +02:00
|
|
|
|
|
|
|
|
|
|
|
class CASViewTestCase(CASTestCase):
|
|
|
|
|
|
|
|
def assertLoginSuccess(self, response, redirect_to=None):
|
|
|
|
"""
|
|
|
|
Asserts response corresponds to a successful login.
|
|
|
|
|
|
|
|
To check this, the response should redirect to redirect_to (default to
|
|
|
|
/accounts/profile/, the default redirect after a successful login).
|
|
|
|
Also CAS_PROVIDER_SESSION_KEY should be set in the client' session. By
|
|
|
|
default, self.client is used.
|
|
|
|
"""
|
|
|
|
if redirect_to is None:
|
|
|
|
redirect_to = settings.LOGIN_REDIRECT_URL
|
|
|
|
|
|
|
|
self.assertRedirects(
|
|
|
|
response, redirect_to,
|
|
|
|
fetch_redirect_response=False,
|
|
|
|
)
|
|
|
|
self.assertIn(
|
|
|
|
CAS_PROVIDER_SESSION_KEY,
|
|
|
|
response.wsgi_request.session,
|
|
|
|
)
|
|
|
|
|
|
|
|
def assertLoginFailure(self, response):
|
|
|
|
"""
|
|
|
|
Asserts response corresponds to a failed login.
|
|
|
|
"""
|
|
|
|
return self.assertInHTML(
|
|
|
|
'<h1>Social Network Login Failure</h1>',
|
|
|
|
str(response.content),
|
|
|
|
)
|