123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518 |
- # coding: utf-8
- """
- 39. Testing using the Test Client
- The test client is a class that can act like a simple
- browser for testing purposes.
- It allows the user to compose GET and POST requests, and
- obtain the response that the server gave to those requests.
- The server Response objects are annotated with the details
- of the contexts and templates that were rendered during the
- process of serving the request.
- ``Client`` objects are stateful - they will retain cookie (and
- thus session) details for the lifetime of the ``Client`` instance.
- This is not intended as a replacement for Twill, Selenium, or
- other browser automation frameworks - it is here to allow
- testing against the contexts and templates produced by a view,
- rather than the HTML rendered to the end-user.
- """
- from __future__ import unicode_literals
- from django.conf import settings
- from django.core import mail
- from django.test import Client, TestCase, RequestFactory
- from django.test.utils import override_settings
- from .views import get_view
- @override_settings(PASSWORD_HASHERS=('django.contrib.auth.hashers.SHA1PasswordHasher',))
- class ClientTest(TestCase):
- fixtures = ['testdata.json']
- def test_get_view(self):
- "GET a view"
- # The data is ignored, but let's check it doesn't crash the system
- # anyway.
- data = {'var': '\xf2'}
- response = self.client.get('/test_client/get_view/', data)
- # Check some response details
- self.assertContains(response, 'This is a test')
- self.assertEqual(response.context['var'], '\xf2')
- self.assertEqual(response.templates[0].name, 'GET Template')
- def test_get_post_view(self):
- "GET a view that normally expects POSTs"
- response = self.client.get('/test_client/post_view/', {})
- # Check some response details
- self.assertEqual(response.status_code, 200)
- self.assertEqual(response.templates[0].name, 'Empty GET Template')
- self.assertTemplateUsed(response, 'Empty GET Template')
- self.assertTemplateNotUsed(response, 'Empty POST Template')
- def test_empty_post(self):
- "POST an empty dictionary to a view"
- response = self.client.post('/test_client/post_view/', {})
- # Check some response details
- self.assertEqual(response.status_code, 200)
- self.assertEqual(response.templates[0].name, 'Empty POST Template')
- self.assertTemplateNotUsed(response, 'Empty GET Template')
- self.assertTemplateUsed(response, 'Empty POST Template')
- def test_post(self):
- "POST some data to a view"
- post_data = {
- 'value': 37
- }
- response = self.client.post('/test_client/post_view/', post_data)
- # Check some response details
- self.assertEqual(response.status_code, 200)
- self.assertEqual(response.context['data'], '37')
- self.assertEqual(response.templates[0].name, 'POST Template')
- self.assertContains(response, 'Data received')
- def test_response_headers(self):
- "Check the value of HTTP headers returned in a response"
- response = self.client.get("/test_client/header_view/")
- self.assertEqual(response['X-DJANGO-TEST'], 'Slartibartfast')
- def test_raw_post(self):
- "POST raw data (with a content type) to a view"
- test_doc = """<?xml version="1.0" encoding="utf-8"?><library><book><title>Blink</title><author>Malcolm Gladwell</author></book></library>"""
- response = self.client.post("/test_client/raw_post_view/", test_doc,
- content_type="text/xml")
- self.assertEqual(response.status_code, 200)
- self.assertEqual(response.templates[0].name, "Book template")
- self.assertEqual(response.content, b"Blink - Malcolm Gladwell")
- def test_insecure(self):
- "GET a URL through http"
- response = self.client.get('/test_client/secure_view/', secure=False)
- self.assertFalse(response.test_was_secure_request)
- self.assertEqual(response.test_server_port, '80')
- def test_secure(self):
- "GET a URL through https"
- response = self.client.get('/test_client/secure_view/', secure=True)
- self.assertTrue(response.test_was_secure_request)
- self.assertEqual(response.test_server_port, '443')
- def test_redirect(self):
- "GET a URL that redirects elsewhere"
- response = self.client.get('/test_client/redirect_view/')
- # Check that the response was a 302 (redirect) and that
- # assertRedirect() understands to put an implicit http://testserver/ in
- # front of non-absolute URLs.
- self.assertRedirects(response, '/test_client/get_view/')
- host = 'django.testserver'
- client_providing_host = Client(HTTP_HOST=host)
- response = client_providing_host.get('/test_client/redirect_view/')
- # Check that the response was a 302 (redirect) with absolute URI
- self.assertRedirects(response, '/test_client/get_view/', host=host)
- def test_redirect_with_query(self):
- "GET a URL that redirects with given GET parameters"
- response = self.client.get('/test_client/redirect_view/', {'var': 'value'})
- # Check if parameters are intact
- self.assertRedirects(response, 'http://testserver/test_client/get_view/?var=value')
- def test_permanent_redirect(self):
- "GET a URL that redirects permanently elsewhere"
- response = self.client.get('/test_client/permanent_redirect_view/')
- # Check that the response was a 301 (permanent redirect)
- self.assertRedirects(response, 'http://testserver/test_client/get_view/', status_code=301)
- client_providing_host = Client(HTTP_HOST='django.testserver')
- response = client_providing_host.get('/test_client/permanent_redirect_view/')
- # Check that the response was a 301 (permanent redirect) with absolute URI
- self.assertRedirects(response, 'http://django.testserver/test_client/get_view/', status_code=301)
- def test_temporary_redirect(self):
- "GET a URL that does a non-permanent redirect"
- response = self.client.get('/test_client/temporary_redirect_view/')
- # Check that the response was a 302 (non-permanent redirect)
- self.assertRedirects(response, 'http://testserver/test_client/get_view/', status_code=302)
- def test_redirect_to_strange_location(self):
- "GET a URL that redirects to a non-200 page"
- response = self.client.get('/test_client/double_redirect_view/')
- # Check that the response was a 302, and that
- # the attempt to get the redirection location returned 301 when retrieved
- self.assertRedirects(response, 'http://testserver/test_client/permanent_redirect_view/', target_status_code=301)
- def test_follow_redirect(self):
- "A URL that redirects can be followed to termination."
- response = self.client.get('/test_client/double_redirect_view/', follow=True)
- self.assertRedirects(response, 'http://testserver/test_client/get_view/', status_code=302, target_status_code=200)
- self.assertEqual(len(response.redirect_chain), 2)
- def test_redirect_http(self):
- "GET a URL that redirects to an http URI"
- response = self.client.get('/test_client/http_redirect_view/', follow=True)
- self.assertFalse(response.test_was_secure_request)
- def test_redirect_https(self):
- "GET a URL that redirects to an https URI"
- response = self.client.get('/test_client/https_redirect_view/', follow=True)
- self.assertTrue(response.test_was_secure_request)
- def test_notfound_response(self):
- "GET a URL that responds as '404:Not Found'"
- response = self.client.get('/test_client/bad_view/')
- # Check that the response was a 404, and that the content contains MAGIC
- self.assertContains(response, 'MAGIC', status_code=404)
- def test_valid_form(self):
- "POST valid data to a form"
- post_data = {
- 'text': 'Hello World',
- 'email': 'foo@example.com',
- 'value': 37,
- 'single': 'b',
- 'multi': ('b', 'c', 'e')
- }
- response = self.client.post('/test_client/form_view/', post_data)
- self.assertEqual(response.status_code, 200)
- self.assertTemplateUsed(response, "Valid POST Template")
- def test_valid_form_with_hints(self):
- "GET a form, providing hints in the GET data"
- hints = {
- 'text': 'Hello World',
- 'multi': ('b', 'c', 'e')
- }
- response = self.client.get('/test_client/form_view/', data=hints)
- self.assertEqual(response.status_code, 200)
- self.assertTemplateUsed(response, "Form GET Template")
- # Check that the multi-value data has been rolled out ok
- self.assertContains(response, 'Select a valid choice.', 0)
- def test_incomplete_data_form(self):
- "POST incomplete data to a form"
- post_data = {
- 'text': 'Hello World',
- 'value': 37
- }
- response = self.client.post('/test_client/form_view/', post_data)
- self.assertContains(response, 'This field is required.', 3)
- self.assertEqual(response.status_code, 200)
- self.assertTemplateUsed(response, "Invalid POST Template")
- self.assertFormError(response, 'form', 'email', 'This field is required.')
- self.assertFormError(response, 'form', 'single', 'This field is required.')
- self.assertFormError(response, 'form', 'multi', 'This field is required.')
- def test_form_error(self):
- "POST erroneous data to a form"
- post_data = {
- 'text': 'Hello World',
- 'email': 'not an email address',
- 'value': 37,
- 'single': 'b',
- 'multi': ('b', 'c', 'e')
- }
- response = self.client.post('/test_client/form_view/', post_data)
- self.assertEqual(response.status_code, 200)
- self.assertTemplateUsed(response, "Invalid POST Template")
- self.assertFormError(response, 'form', 'email', 'Enter a valid email address.')
- def test_valid_form_with_template(self):
- "POST valid data to a form using multiple templates"
- post_data = {
- 'text': 'Hello World',
- 'email': 'foo@example.com',
- 'value': 37,
- 'single': 'b',
- 'multi': ('b', 'c', 'e')
- }
- response = self.client.post('/test_client/form_view_with_template/', post_data)
- self.assertContains(response, 'POST data OK')
- self.assertTemplateUsed(response, "form_view.html")
- self.assertTemplateUsed(response, 'base.html')
- self.assertTemplateNotUsed(response, "Valid POST Template")
- def test_incomplete_data_form_with_template(self):
- "POST incomplete data to a form using multiple templates"
- post_data = {
- 'text': 'Hello World',
- 'value': 37
- }
- response = self.client.post('/test_client/form_view_with_template/', post_data)
- self.assertContains(response, 'POST data has errors')
- self.assertTemplateUsed(response, 'form_view.html')
- self.assertTemplateUsed(response, 'base.html')
- self.assertTemplateNotUsed(response, "Invalid POST Template")
- self.assertFormError(response, 'form', 'email', 'This field is required.')
- self.assertFormError(response, 'form', 'single', 'This field is required.')
- self.assertFormError(response, 'form', 'multi', 'This field is required.')
- def test_form_error_with_template(self):
- "POST erroneous data to a form using multiple templates"
- post_data = {
- 'text': 'Hello World',
- 'email': 'not an email address',
- 'value': 37,
- 'single': 'b',
- 'multi': ('b', 'c', 'e')
- }
- response = self.client.post('/test_client/form_view_with_template/', post_data)
- self.assertContains(response, 'POST data has errors')
- self.assertTemplateUsed(response, "form_view.html")
- self.assertTemplateUsed(response, 'base.html')
- self.assertTemplateNotUsed(response, "Invalid POST Template")
- self.assertFormError(response, 'form', 'email', 'Enter a valid email address.')
- def test_unknown_page(self):
- "GET an invalid URL"
- response = self.client.get('/test_client/unknown_view/')
- # Check that the response was a 404
- self.assertEqual(response.status_code, 404)
- def test_url_parameters(self):
- "Make sure that URL ;-parameters are not stripped."
- response = self.client.get('/test_client/unknown_view/;some-parameter')
- # Check that the path in the response includes it (ignore that it's a 404)
- self.assertEqual(response.request['PATH_INFO'], '/test_client/unknown_view/;some-parameter')
- def test_view_with_login(self):
- "Request a page that is protected with @login_required"
- # Get the page without logging in. Should result in 302.
- response = self.client.get('/test_client/login_protected_view/')
- self.assertRedirects(response, 'http://testserver/accounts/login/?next=/test_client/login_protected_view/')
- # Log in
- login = self.client.login(username='testclient', password='password')
- self.assertTrue(login, 'Could not log in')
- # Request a page that requires a login
- response = self.client.get('/test_client/login_protected_view/')
- self.assertEqual(response.status_code, 200)
- self.assertEqual(response.context['user'].username, 'testclient')
- def test_view_with_method_login(self):
- "Request a page that is protected with a @login_required method"
- # Get the page without logging in. Should result in 302.
- response = self.client.get('/test_client/login_protected_method_view/')
- self.assertRedirects(response, 'http://testserver/accounts/login/?next=/test_client/login_protected_method_view/')
- # Log in
- login = self.client.login(username='testclient', password='password')
- self.assertTrue(login, 'Could not log in')
- # Request a page that requires a login
- response = self.client.get('/test_client/login_protected_method_view/')
- self.assertEqual(response.status_code, 200)
- self.assertEqual(response.context['user'].username, 'testclient')
- def test_view_with_login_and_custom_redirect(self):
- "Request a page that is protected with @login_required(redirect_field_name='redirect_to')"
- # Get the page without logging in. Should result in 302.
- response = self.client.get('/test_client/login_protected_view_custom_redirect/')
- self.assertRedirects(response, 'http://testserver/accounts/login/?redirect_to=/test_client/login_protected_view_custom_redirect/')
- # Log in
- login = self.client.login(username='testclient', password='password')
- self.assertTrue(login, 'Could not log in')
- # Request a page that requires a login
- response = self.client.get('/test_client/login_protected_view_custom_redirect/')
- self.assertEqual(response.status_code, 200)
- self.assertEqual(response.context['user'].username, 'testclient')
- def test_view_with_bad_login(self):
- "Request a page that is protected with @login, but use bad credentials"
- login = self.client.login(username='otheruser', password='nopassword')
- self.assertFalse(login)
- def test_view_with_inactive_login(self):
- "Request a page that is protected with @login, but use an inactive login"
- login = self.client.login(username='inactive', password='password')
- self.assertFalse(login)
- def test_logout(self):
- "Request a logout after logging in"
- # Log in
- self.client.login(username='testclient', password='password')
- # Request a page that requires a login
- response = self.client.get('/test_client/login_protected_view/')
- self.assertEqual(response.status_code, 200)
- self.assertEqual(response.context['user'].username, 'testclient')
- # Log out
- self.client.logout()
- # Request a page that requires a login
- response = self.client.get('/test_client/login_protected_view/')
- self.assertRedirects(response, 'http://testserver/accounts/login/?next=/test_client/login_protected_view/')
- def test_view_with_permissions(self):
- "Request a page that is protected with @permission_required"
- # Get the page without logging in. Should result in 302.
- response = self.client.get('/test_client/permission_protected_view/')
- self.assertRedirects(response, 'http://testserver/accounts/login/?next=/test_client/permission_protected_view/')
- # Log in
- login = self.client.login(username='testclient', password='password')
- self.assertTrue(login, 'Could not log in')
- # Log in with wrong permissions. Should result in 302.
- response = self.client.get('/test_client/permission_protected_view/')
- self.assertRedirects(response, 'http://testserver/accounts/login/?next=/test_client/permission_protected_view/')
- # TODO: Log in with right permissions and request the page again
- def test_view_with_permissions_exception(self):
- "Request a page that is protected with @permission_required but raises a exception"
- # Get the page without logging in. Should result in 403.
- response = self.client.get('/test_client/permission_protected_view_exception/')
- self.assertEqual(response.status_code, 403)
- # Log in
- login = self.client.login(username='testclient', password='password')
- self.assertTrue(login, 'Could not log in')
- # Log in with wrong permissions. Should result in 403.
- response = self.client.get('/test_client/permission_protected_view_exception/')
- self.assertEqual(response.status_code, 403)
- def test_view_with_method_permissions(self):
- "Request a page that is protected with a @permission_required method"
- # Get the page without logging in. Should result in 302.
- response = self.client.get('/test_client/permission_protected_method_view/')
- self.assertRedirects(response, 'http://testserver/accounts/login/?next=/test_client/permission_protected_method_view/')
- # Log in
- login = self.client.login(username='testclient', password='password')
- self.assertTrue(login, 'Could not log in')
- # Log in with wrong permissions. Should result in 302.
- response = self.client.get('/test_client/permission_protected_method_view/')
- self.assertRedirects(response, 'http://testserver/accounts/login/?next=/test_client/permission_protected_method_view/')
- # TODO: Log in with right permissions and request the page again
- def test_external_redirect(self):
- response = self.client.get('/test_client/django_project_redirect/')
- self.assertRedirects(response, 'https://www.djangoproject.com/', fetch_redirect_response=False)
- def test_session_modifying_view(self):
- "Request a page that modifies the session"
- # Session value isn't set initially
- try:
- self.client.session['tobacconist']
- self.fail("Shouldn't have a session value")
- except KeyError:
- pass
- from django.contrib.sessions.models import Session
- self.client.post('/test_client/session_view/')
- # Check that the session was modified
- self.assertEqual(self.client.session['tobacconist'], 'hovercraft')
- def test_view_with_exception(self):
- "Request a page that is known to throw an error"
- self.assertRaises(KeyError, self.client.get, "/test_client/broken_view/")
- #Try the same assertion, a different way
- try:
- self.client.get('/test_client/broken_view/')
- self.fail('Should raise an error')
- except KeyError:
- pass
- def test_mail_sending(self):
- "Test that mail is redirected to a dummy outbox during test setup"
- response = self.client.get('/test_client/mail_sending_view/')
- self.assertEqual(response.status_code, 200)
- self.assertEqual(len(mail.outbox), 1)
- self.assertEqual(mail.outbox[0].subject, 'Test message')
- self.assertEqual(mail.outbox[0].body, 'This is a test email')
- self.assertEqual(mail.outbox[0].from_email, 'from@example.com')
- self.assertEqual(mail.outbox[0].to[0], 'first@example.com')
- self.assertEqual(mail.outbox[0].to[1], 'second@example.com')
- def test_mass_mail_sending(self):
- "Test that mass mail is redirected to a dummy outbox during test setup"
- response = self.client.get('/test_client/mass_mail_sending_view/')
- self.assertEqual(response.status_code, 200)
- self.assertEqual(len(mail.outbox), 2)
- self.assertEqual(mail.outbox[0].subject, 'First Test message')
- self.assertEqual(mail.outbox[0].body, 'This is the first test email')
- self.assertEqual(mail.outbox[0].from_email, 'from@example.com')
- self.assertEqual(mail.outbox[0].to[0], 'first@example.com')
- self.assertEqual(mail.outbox[0].to[1], 'second@example.com')
- self.assertEqual(mail.outbox[1].subject, 'Second Test message')
- self.assertEqual(mail.outbox[1].body, 'This is the second test email')
- self.assertEqual(mail.outbox[1].from_email, 'from@example.com')
- self.assertEqual(mail.outbox[1].to[0], 'second@example.com')
- self.assertEqual(mail.outbox[1].to[1], 'third@example.com')
- @override_settings(
- MIDDLEWARE_CLASSES = ('django.middleware.csrf.CsrfViewMiddleware',)
- )
- class CSRFEnabledClientTests(TestCase):
- def test_csrf_enabled_client(self):
- "A client can be instantiated with CSRF checks enabled"
- csrf_client = Client(enforce_csrf_checks=True)
- # The normal client allows the post
- response = self.client.post('/test_client/post_view/', {})
- self.assertEqual(response.status_code, 200)
- # The CSRF-enabled client rejects it
- response = csrf_client.post('/test_client/post_view/', {})
- self.assertEqual(response.status_code, 403)
- class CustomTestClient(Client):
- i_am_customized = "Yes"
- class CustomTestClientTest(TestCase):
- client_class = CustomTestClient
- def test_custom_test_client(self):
- """A test case can specify a custom class for self.client."""
- self.assertEqual(hasattr(self.client, "i_am_customized"), True)
- class RequestFactoryTest(TestCase):
- def test_request_factory(self):
- factory = RequestFactory()
- request = factory.get('/somewhere/')
- response = get_view(request)
- self.assertEqual(response.status_code, 200)
- self.assertContains(response, 'This is a test')
|