-
Notifications
You must be signed in to change notification settings - Fork 3.9k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
test: add validation tests and document how to test views
- Loading branch information
1 parent
170a438
commit 6f6f130
Showing
2 changed files
with
114 additions
and
23 deletions.
There are no files selected for viewing
60 changes: 60 additions & 0 deletions
60
cms/djangoapps/contentstore/docs/how-tos/test_course_related_view_auth.rst
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,60 @@ | ||
============================================== | ||
How to test View Auth for course-related views | ||
============================================== | ||
|
||
What to test | ||
------------ | ||
Each view endpoint that exposes an internal API endpoint - like in files in the rest_api folder - must | ||
be tested for the following. | ||
|
||
- Only authenticated users can access the endpoint. | ||
- Only users with the correct permissions (authorization) can access the endpoint. | ||
- All data and params that are part of the request are properly validated. | ||
|
||
How to test | ||
----------- | ||
A lot of these tests can be easily implemented by inheriting from the `AuthorizeStaffTestCase`. | ||
This parent class assumes that the view is for a specific course and that only users who have access | ||
to the course can access the view. (They are either staff or instructors for the course, or global admin). | ||
|
||
Here is an example of how to test a view that requires a user to be authenticated and have access to a course. | ||
|
||
.. code-block:: python | ||
from cms.djangoapps.contentstore.tests.test_utils import AuthorizeStaffTestCase | ||
from django.test import TestCase | ||
from xmodule.modulestore.tests.django_utils import ModuleStoreTestCase | ||
from django.urls import reverse | ||
class TestMyGetView(AuthorizeStaffTestCase, ModuleStoreTestCase, TestCase): | ||
def make_request(self, course_id=None, data=None): | ||
url = self.get_url(self.course.id) | ||
response = self.client.get(url, data) | ||
return response | ||
def get_url(self, course_key): | ||
url = reverse( | ||
'cms.djangoapps.contentstore:v0:my_get_view', | ||
kwargs={'course_id': self.course.id} | ||
) | ||
return url | ||
As you can see, you need to inherit from `AuthorizeStaffTestCase` and `ModuleStoreTestCase`, and then either | ||
`TestCase` or `APITestCase` depending on the type of view you are testing. For cookie-based | ||
authentication, `TestCase` is sufficient, for Oauth2 use `ApiTestCase`. | ||
|
||
The only two methods you need to implement are `make_request` and `get_url`. The `make_request` method | ||
should make the request to the view and return the response. The `get_url` method should return the URL | ||
for the view you are testing. | ||
|
||
Overwriting Tests | ||
----------------- | ||
If you need different behavior you can overwrite the tests from the parent class. | ||
For example, if students should have access to the view, simply implement the | ||
`test_student` method in your test class. | ||
|
||
Adding other tests | ||
------------------ | ||
If you want to test other things in the view - let's say validation - | ||
it's easy to just add another `test_...` function to your test class | ||
and you can use the `make_request` method to make the request. |
77 changes: 54 additions & 23 deletions
77
cms/djangoapps/contentstore/rest_api/v0/tests/test_course_optimizer.py
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,42 +1,73 @@ | ||
from cms.djangoapps.contentstore.tests.test_utils import AuthorizeStaffTestCase | ||
from common.djangoapps.student.tests.factories import InstructorFactory | ||
from rest_framework import status | ||
from django.test import TestCase | ||
from xmodule.modulestore.tests.django_utils import ModuleStoreTestCase | ||
from django.urls import reverse | ||
|
||
class TestCourseOptimizer(AuthorizeStaffTestCase, ModuleStoreTestCase, TestCase): | ||
class TestGetLinkCheckStatus(AuthorizeStaffTestCase, ModuleStoreTestCase, TestCase): | ||
''' | ||
Tests for CourseOptimizer | ||
Authentication and Authorization Tests for CourseOptimizer. | ||
For concrete tests that are run, check `AuthorizeStaffTestCase`. | ||
''' | ||
def test_inherited(self): | ||
# This method ensures that pytest recognizes this class as containing tests | ||
pass | ||
|
||
def make_request(self, course_id=None, data=None): | ||
def make_request(self, course_id=None, data=None, **kwargs): | ||
url = self.get_url(self.course.id) | ||
print('make_request url: ', url) | ||
response = self.client.get(url, data) | ||
print('make_request response status code: ', response.status_code) | ||
print('make_request response content: ', response.content) | ||
return response | ||
|
||
def get_url(self, course_key): | ||
url = reverse( | ||
'cms.djangoapps.contentstore:v0:link_check_status', | ||
kwargs={'course_id': self.course.id} | ||
) | ||
print('get_url: ', url) | ||
return url | ||
|
||
def test_produces_4xx_when_invalid_course_id(self): | ||
''' | ||
Test course_id validation | ||
''' | ||
response = self.make_request(course_id='invalid_course_id') | ||
self.assertIn(response.status_code, range(400, 500)) | ||
|
||
def test_produces_4xx_when_additional_kwargs(self): | ||
''' | ||
Test additional kwargs validation | ||
''' | ||
response = self.make_request(course_id=self.course.id, malicious_kwarg='malicious_kwarg') | ||
self.assertIn(response.status_code, range(400, 500)) | ||
|
||
def test_course_instructor(self, expect_status=status.HTTP_200_OK): | ||
self.course_instructor = InstructorFactory( | ||
username='instructor', | ||
password=self.password, | ||
course_key=self.course.id, | ||
) | ||
self.client.login(username=self.course_instructor.username, password=self.password) | ||
response = self.make_request() | ||
print('test_course_instructor response status code: ', response.status_code) | ||
assert response.status_code == expect_status | ||
class TestPostLinkCheck(AuthorizeStaffTestCase, ModuleStoreTestCase, TestCase): | ||
''' | ||
Authentication and Authorization Tests for CourseOptimizer. | ||
For concrete tests that are run, check `AuthorizeStaffTestCase`. | ||
''' | ||
def make_request(self, course_id=None, data=None, **kwargs): | ||
url = self.get_url(self.course.id) | ||
response = self.client.post(url, data) | ||
return response | ||
|
||
def get_url(self, course_key): | ||
url = reverse( | ||
'cms.djangoapps.contentstore:v0:link_check', | ||
kwargs={'course_id': self.course.id} | ||
) | ||
return url | ||
|
||
def test_produces_4xx_when_invalid_course_id(self): | ||
''' | ||
Test course_id validation | ||
''' | ||
response = self.make_request(course_id='invalid_course_id') | ||
self.assertIn(response.status_code, range(400, 500)) | ||
|
||
def test_produces_4xx_when_additional_kwargs(self): | ||
''' | ||
Test additional kwargs validation | ||
''' | ||
response = self.make_request(course_id=self.course.id, malicious_kwarg='malicious_kwarg') | ||
self.assertIn(response.status_code, range(400, 500)) | ||
|
||
def test_produces_4xx_when_unexpected_data(self): | ||
''' | ||
Test validation when request contains unexpected data | ||
''' | ||
response = self.make_request(course_id=self.course.id, data={'unexpected_data': 'unexpected_data'}) | ||
self.assertIn(response.status_code, range(400, 500)) |