Skip to content

Commit

Permalink
tests for training/courses
Browse files Browse the repository at this point in the history
  • Loading branch information
superryeti committed Apr 13, 2023
1 parent b1cf22f commit 905f170
Showing 1 changed file with 282 additions and 0 deletions.
282 changes: 282 additions & 0 deletions physionet-django/training/test_views.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,282 @@
import os
import json
import shutil

from django.conf import settings

from lightwave.views import DBCAL_FILE, ORIGINAL_DBCAL_FILE
from django.contrib.messages.storage.fallback import FallbackStorage
from django.core.files.uploadedfile import SimpleUploadedFile
from django.test import TestCase
from django.urls import reverse
from training.models import Course, ContentBlock, Quiz, QuizChoice
from user.models import Training


def _force_delete_tree(path):
"""
Recursively delete a directory tree, including read-only directories.
"""
if os.path.exists(path):
# Make each (recursive) subdirectory writable, so that we can
# delete files from it.
for subdir, _, _ in os.walk(path):
os.chmod(subdir, 0o700)
shutil.rmtree(path)


class TestMixin(TestCase):
"""
Mixin for test methods
Because the fixtures are installed and database is rolled back
before each setup and teardown respectively, the demo test files
will be created and destroyed after each test also. We want the
demo files as well as the demo data reset each time, and individual
test methods such as publishing projects may change the files.
Note about inheriting: https://nedbatchelder.com/blog/201210/multiple_inheritance_is_hard.html
"""

def setUp(self):
"""
Copy demo media files to the testing media root.
Copy demo static files to the testing effective static root.
Symlink dbcal file to the testing effective static root.
Does not run collectstatic. The StaticLiveServerTestCase should
do that automatically for tests that need it.
"""
_force_delete_tree(settings.MEDIA_ROOT)
shutil.copytree(os.path.abspath(os.path.join(settings.DEMO_FILE_ROOT, 'media')),
settings.MEDIA_ROOT)

self.test_static_root = settings.STATIC_ROOT if settings.STATIC_ROOT else settings.STATICFILES_DIRS[0]
_force_delete_tree(self.test_static_root)
shutil.copytree(os.path.abspath(os.path.join(settings.DEMO_FILE_ROOT, 'static')),
self.test_static_root)

if os.path.exists(ORIGINAL_DBCAL_FILE):
os.symlink(ORIGINAL_DBCAL_FILE, DBCAL_FILE)

# Published project files should have been made read-only at
# the time of publication
for topdir in (settings.MEDIA_ROOT, self.test_static_root):
ppdir = os.path.join(topdir, 'published-projects')
for dirpath, subdirs, files in os.walk(ppdir):
if dirpath != ppdir:
for f in files:
os.chmod(os.path.join(dirpath, f), 0o444)
for d in subdirs:
os.chmod(os.path.join(dirpath, d), 0o555)

def tearDown(self):
"""
Remove the testing media root
"""
_force_delete_tree(settings.MEDIA_ROOT)
_force_delete_tree(self.test_static_root)

def assertMessage(self, response, level):
"""
Assert that the max message level in the request equals `level`.
Can use message success or error to test outcome, since there
are different cases where forms are reloaded, not present, etc.
The response code for invalid form submissions are still 200
so cannot use that to test form submissions.
"""
self.assertEqual(max(m.level for m in response.context['messages']),
level)

def make_get_request(self, viewname, reverse_kwargs=None):
"""
Helper Function.
Create and set a get request
- viewname: The view name
- reverse_kwargs: kwargs of additional url parameters
"""
self.get_request = self.factory.get(reverse(viewname,
kwargs=reverse_kwargs))
self.get_request.user = self.user

def make_post_request(self, viewname, data, reverse_kwargs=None):
"""
Helper Function.
Create and set a get request
- viewname: The view name
- data: Dictionary of post parameters
- reverse_kwargs: Kwargs of additional url parameters
"""
self.post_request = self.factory.post(reverse(viewname,
kwargs=reverse_kwargs), data)
self.post_request.user = self.user
# Provide the message object to the request because middleware
# is not supported by RequestFactory
setattr(self.post_request, 'session', 'session')
messages = FallbackStorage(self.post_request)
setattr(self.post_request, '_messages', messages)

def tst_get_request(self, view, view_kwargs=None, status_code=200,
redirect_viewname=None, redirect_reverse_kwargs=None):
"""
Helper Function.
Test the get request with the view against the expected status code
- view: The view function
- view_kwargs: The kwargs dictionary of additional arguments to put into
view function aside from request
- status_code: expected status code of response
- redirect_viewname: view name of the expected redirect
- redirect_reverse_kwargs: kwargs dictionary of expected redirect
"""
if view_kwargs:
response = view(self.get_request, **view_kwargs)
else:
response = view(self.get_request)
self.assertEqual(response.status_code, status_code)
if status_code == 302:
# We don't use assertRedirects because the response has no client
self.assertEqual(response['location'], reverse(redirect_viewname,
kwargs=redirect_reverse_kwargs))

def tst_post_request(self, view, view_kwargs=None, status_code=200,
redirect_viewname=None, redirect_reverse_kwargs=None):
"""
Helper Function.
Test the post request with the view against the expected status code
- view: The view function
- view_kwargs: The kwargs dictionary of additional arguments to put into
view function aside from request
- status_code: expected status code of response
- redirect_viewname: view name of the expected redirect
- redirect_reverse_kwargs: kwargs dictionary of expected redirect
"""
if view_kwargs:
response = view(self.post_request, **view_kwargs)
else:
response = view(self.post_request)
self.assertEqual(response.status_code, status_code)
if status_code == 302:
self.assertEqual(response['location'], reverse(redirect_viewname,
kwargs=redirect_reverse_kwargs))


class TestPlatformTraining(TestMixin):
""" Test that all views are behaving as expected """

def setUp(self):
"""Setup for tests"""

super().setUp()
self.client.login(username='admin', password='Tester11!')

def test_take_training_get(self):
"""test if the training page loads"""

response = self.client.get(reverse("platform_training", kwargs={'training_id': 2}))
self.assertEqual(response.status_code, 200)
self.assertTemplateUsed(response, "training/course.html")

def test_take_training_post_redirect_valid(self):
"""test if the post request to training redirects to the correct page"""

response = self.client.post(reverse("start_training"), {"training_type": 2})
self.assertRedirects(response, reverse("platform_training",
kwargs={'training_id': 2}), status_code=302)

def test_take_training_quiz_post_valid(self):
"""test the views needed for a user to take a training course"""
question_answers_module1 = '{"1":3,"2":5,"3":12,"4":16}'
question_answers_module2 = '{"5":18,"6":22}'
trainings = Training.objects.count()

# access the module 1
self.client.get(reverse("platform_training_module", kwargs={'training_id': 2, 'module_id': 1}))
# simulate the post request of clicking the next button to start tracking
self.client.post(
reverse("update_module_progress"),
data={'course_id': 1, 'module_id': 1, 'update_type': 'content', 'update_type_id': 1}
)
# submit the module 1
response1 = self.client.post(
reverse("platform_training_module", kwargs={'training_id': 2, 'module_id': 1}),
data={
'question_answers': question_answers_module1
}
)
self.assertRedirects(response1, reverse("platform_training", kwargs={'training_id': 2}), status_code=302)

# access the module 2
self.client.get(reverse("platform_training_module", kwargs={'training_id': 2, 'module_id': 1}))
# simulate the post request of clicking the next button to start tracking
self.client.post(
reverse("update_module_progress"),
data={'course_id': 1, 'module_id': 2, 'update_type': 'content', 'update_type_id': 5}
)
# submit the module 2
response2 = self.client.post(
reverse("platform_training_module", kwargs={'training_id': 2, 'module_id': 2}),
data={
'question_answers': question_answers_module2
}
)
self.assertRedirects(response2, reverse("platform_training", kwargs={'training_id': 2}), status_code=302)
self.assertEqual(Training.objects.count(), trainings + 1)

def test_create_course_get(self):
"""test if admin can access the courses page"""

response = self.client.get(reverse("courses"))
self.assertEqual(response.status_code, 200)
self.assertTemplateUsed(response, "console/training_type/index.html")

def test_create_course_post_valid(self):
"""test the post request to create a new course"""

file_path = os.path.join(settings.BASE_DIR, "static", "sample", "example-course-create.json")
with open(file_path, 'r') as f:
content = f.read()
content_json = json.loads(content)
response = self.client.post(
reverse("courses"),
data={
"training_id": -1,
"create": ['Submit'],
"json_file": SimpleUploadedFile(f.name, content.encode()),
}
)
self.assertRedirects(response, reverse("courses"), status_code=302)
results = Course.objects.filter(training_type__name=content_json['name'])
self.assertEqual(results.count(), 1)
return results.first()

def test_update_course_post_valid(self):
"""test the post request to update a course"""

# create a training
training = self.test_create_course_post_valid()

file_path = os.path.join(settings.BASE_DIR, "static", "sample", "example-course-update.json")
with open(file_path, 'r') as f:
content = f.read()
content_json = json.loads(content)
response = self.client.post(
reverse("courses"),
data={
"training_id": training.training_type_id,
"update": ['Submit'],
"json_file": SimpleUploadedFile(f.name, content.encode()),
}
)

self.assertRedirects(response, reverse("courses"), status_code=302)
results = Course.objects.filter(training_type__name=content_json['name'])
self.assertEqual(results.count(), 2)
self.assertEqual(results.last().version, float(content_json['courses'][0]['version']))

0 comments on commit 905f170

Please sign in to comment.