Commit f5944c2e by Dillon Dumesnil

Improving performance of tests by creating everything but the review section…

Improving performance of tests by creating everything but the review section prior to running tests and then creating the review section, unit, and XBlock with each test
parent 2a68f9b1
...@@ -11,14 +11,14 @@ from nose.plugins.attrib import attr ...@@ -11,14 +11,14 @@ from nose.plugins.attrib import attr
from lms.djangoapps.courseware.tests.factories import GlobalStaffFactory from lms.djangoapps.courseware.tests.factories import GlobalStaffFactory
from lms.djangoapps.courseware.tests.helpers import LoginEnrollmentTestCase from lms.djangoapps.courseware.tests.helpers import LoginEnrollmentTestCase
from xmodule.modulestore.tests.django_utils import ModuleStoreTestCase from xmodule.modulestore.tests.django_utils import SharedModuleStoreTestCase
from xmodule.modulestore.tests.factories import CourseFactory, ItemFactory from xmodule.modulestore.tests.factories import CourseFactory, ItemFactory
from review import get_review_ids from review import get_review_ids
import crum import crum
class TestReviewXBlock(ModuleStoreTestCase, LoginEnrollmentTestCase): class TestReviewXBlock(SharedModuleStoreTestCase, LoginEnrollmentTestCase):
""" """
Create the test environment with the review xblock. Create the test environment with the review xblock.
""" """
...@@ -27,16 +27,18 @@ class TestReviewXBlock(ModuleStoreTestCase, LoginEnrollmentTestCase): ...@@ -27,16 +27,18 @@ class TestReviewXBlock(ModuleStoreTestCase, LoginEnrollmentTestCase):
] ]
XBLOCK_NAMES = ['review'] XBLOCK_NAMES = ['review']
def setUp(self): @classmethod
super(TestReviewXBlock, self).setUp() def setUpClass(cls):
# Nose runs setUpClass methods even if a class decorator says to skip # Nose runs setUpClass methods even if a class decorator says to skip
# the class: https://github.com/nose-devs/nose/issues/946 # the class: https://github.com/nose-devs/nose/issues/946
# So, skip the test class here if we are not in the LMS. # So, skip the test class here if we are not in the LMS.
if settings.ROOT_URLCONF != 'lms.urls': if settings.ROOT_URLCONF != 'lms.urls':
raise unittest.SkipTest('Test only valid in lms') raise unittest.SkipTest('Test only valid in lms')
super(TestReviewXBlock, cls).setUpClass()
# Set up for the actual course # Set up for the actual course
self.course_actual = CourseFactory.create( cls.course_actual = CourseFactory.create(
display_name='Review_Test_Course_ACTUAL', display_name='Review_Test_Course_ACTUAL',
org='DillonX', org='DillonX',
number='DAD101x', number='DAD101x',
...@@ -44,139 +46,134 @@ class TestReviewXBlock(ModuleStoreTestCase, LoginEnrollmentTestCase): ...@@ -44,139 +46,134 @@ class TestReviewXBlock(ModuleStoreTestCase, LoginEnrollmentTestCase):
) )
# There are multiple sections so the learner can load different # There are multiple sections so the learner can load different
# problems, but should only be shown review problems from what they have loaded # problems, but should only be shown review problems from what they have loaded
with self.store.bulk_operations(self.course_actual.id, emit_signals=False): with cls.store.bulk_operations(cls.course_actual.id, emit_signals=False):
self.chapter_actual = ItemFactory.create( cls.chapter_actual = ItemFactory.create(
parent=self.course_actual, display_name='Overview' parent=cls.course_actual, display_name='Overview'
) )
self.section1_actual = ItemFactory.create( cls.section1_actual = ItemFactory.create(
parent=self.chapter_actual, display_name='Section 1' parent=cls.chapter_actual, display_name='Section 1'
) )
self.unit1_actual = ItemFactory.create( cls.unit1_actual = ItemFactory.create(
parent=self.section1_actual, display_name='New Unit 1' parent=cls.section1_actual, display_name='New Unit 1'
) )
self.xblock1_actual = ItemFactory.create( cls.xblock1_actual = ItemFactory.create(
parent=self.unit1_actual, parent=cls.unit1_actual,
category='problem', category='problem',
display_name='Problem 1' display_name='Problem 1'
) )
self.xblock2_actual = ItemFactory.create( cls.xblock2_actual = ItemFactory.create(
parent=self.unit1_actual, parent=cls.unit1_actual,
category='problem', category='problem',
display_name='Problem 2' display_name='Problem 2'
) )
self.xblock3_actual = ItemFactory.create( cls.xblock3_actual = ItemFactory.create(
parent=self.unit1_actual, parent=cls.unit1_actual,
category='problem', category='problem',
display_name='Problem 3' display_name='Problem 3'
) )
self.xblock4_actual = ItemFactory.create( cls.xblock4_actual = ItemFactory.create(
parent=self.unit1_actual, parent=cls.unit1_actual,
category='problem', category='problem',
display_name='Problem 4' display_name='Problem 4'
) )
self.section2_actual = ItemFactory.create( cls.section2_actual = ItemFactory.create(
parent=self.chapter_actual, display_name='Section 2' parent=cls.chapter_actual, display_name='Section 2'
) )
self.unit2_actual = ItemFactory.create( cls.unit2_actual = ItemFactory.create(
parent=self.section2_actual, display_name='New Unit 2' parent=cls.section2_actual, display_name='New Unit 2'
) )
self.xblock5_actual = ItemFactory.create( cls.xblock5_actual = ItemFactory.create(
parent=self.unit2_actual, parent=cls.unit2_actual,
category='problem', category='problem',
display_name='Problem 5' display_name='Problem 5'
) )
self.section3_actual = ItemFactory.create( cls.section3_actual = ItemFactory.create(
parent=self.chapter_actual, display_name='Section 3' parent=cls.chapter_actual, display_name='Section 3'
) )
self.unit3_actual = ItemFactory.create( cls.unit3_actual = ItemFactory.create(
parent=self.section3_actual, display_name='New Unit 3' parent=cls.section3_actual, display_name='New Unit 3'
) )
self.xblock6_actual = ItemFactory.create( cls.xblock6_actual = ItemFactory.create(
parent=self.unit3_actual, parent=cls.unit3_actual,
category='problem', category='problem',
display_name='Problem 6' display_name='Problem 6'
) )
# This is the actual review xBlock
# When implemented, the review is in its own section as a
# stand-alone unit.
self.review_section_actual = ItemFactory.create(
parent=self.chapter_actual, display_name='Review Subsection'
)
self.review_unit_actual = ItemFactory.create(
parent=self.review_section_actual, display_name='Review Unit'
)
self.course_actual_url = reverse( cls.course_actual_url = reverse(
'courseware_section', 'courseware_section',
kwargs={ kwargs={
'course_id': unicode(self.course_actual.id), 'course_id': unicode(cls.course_actual.id),
'chapter': 'Overview', 'chapter': 'Overview',
'section': 'Welcome', 'section': 'Welcome',
} }
) )
# Set up for the review course where the review problems are hosted # Set up for the review course where the review problems are hosted
self.course_review = CourseFactory.create( cls.course_review = CourseFactory.create(
display_name='Review_Test_Course_REVIEW', display_name='Review_Test_Course_REVIEW',
org='DillonX', org='DillonX',
number='DAD101x_review', number='DAD101x_review',
run='3T2017' run='3T2017'
) )
with self.store.bulk_operations(self.course_review.id, emit_signals=True): with cls.store.bulk_operations(cls.course_review.id, emit_signals=True):
self.chapter_review = ItemFactory.create( cls.chapter_review = ItemFactory.create(
parent=self.course_review, display_name='Overview' parent=cls.course_review, display_name='Overview'
) )
self.section_review = ItemFactory.create( cls.section_review = ItemFactory.create(
parent=self.chapter_review, display_name='Welcome' parent=cls.chapter_review, display_name='Welcome'
) )
self.unit1_review = ItemFactory.create( cls.unit1_review = ItemFactory.create(
parent=self.section_review, display_name='New Unit 1' parent=cls.section_review, display_name='New Unit 1'
) )
self.xblock1_review = ItemFactory.create( cls.xblock1_review = ItemFactory.create(
parent=self.unit1_review, parent=cls.unit1_review,
category='problem', category='problem',
display_name='Problem 1' display_name='Problem 1'
) )
self.xblock2_review = ItemFactory.create( cls.xblock2_review = ItemFactory.create(
parent=self.unit1_review, parent=cls.unit1_review,
category='problem', category='problem',
display_name='Problem 2' display_name='Problem 2'
) )
self.xblock3_review = ItemFactory.create( cls.xblock3_review = ItemFactory.create(
parent=self.unit1_review, parent=cls.unit1_review,
category='problem', category='problem',
display_name='Problem 3' display_name='Problem 3'
) )
self.xblock4_review = ItemFactory.create( cls.xblock4_review = ItemFactory.create(
parent=self.unit1_review, parent=cls.unit1_review,
category='problem', category='problem',
display_name='Problem 4' display_name='Problem 4'
) )
self.unit2_review = ItemFactory.create( cls.unit2_review = ItemFactory.create(
parent=self.section_review, display_name='New Unit 2' parent=cls.section_review, display_name='New Unit 2'
) )
self.xblock5_review = ItemFactory.create( cls.xblock5_review = ItemFactory.create(
parent=self.unit2_review, parent=cls.unit2_review,
category='problem', category='problem',
display_name='Problem 5' display_name='Problem 5'
) )
self.unit3_review = ItemFactory.create( cls.unit3_review = ItemFactory.create(
parent=self.section_review, display_name='New Unit 3' parent=cls.section_review, display_name='New Unit 3'
) )
self.xblock6_review = ItemFactory.create( cls.xblock6_review = ItemFactory.create(
parent=self.unit3_review, parent=cls.unit3_review,
category='problem', category='problem',
display_name='Problem 6' display_name='Problem 6'
) )
self.course_review_url = reverse( cls.course_review_url = reverse(
'courseware_section', 'courseware_section',
kwargs={ kwargs={
'course_id': unicode(self.course_review.id), 'course_id': unicode(cls.course_review.id),
'chapter': 'Overview', 'chapter': 'Overview',
'section': 'Welcome', 'section': 'Welcome',
} }
) )
def setUp(self):
super(TestReviewXBlock, self).setUp()
for idx, student in enumerate(self.STUDENTS): for idx, student in enumerate(self.STUDENTS):
username = 'u{}'.format(idx) username = 'u{}'.format(idx)
self.create_account(username, student['email'], student['password']) self.create_account(username, student['email'], student['password'])
...@@ -209,10 +206,18 @@ class TestReviewFunctions(TestReviewXBlock): ...@@ -209,10 +206,18 @@ class TestReviewFunctions(TestReviewXBlock):
self.enroll_student(self.STUDENTS[0]['email'], self.STUDENTS[0]['password'], self.course_actual) self.enroll_student(self.STUDENTS[0]['email'], self.STUDENTS[0]['password'], self.course_actual)
self.enroll_student(self.STUDENTS[0]['email'], self.STUDENTS[0]['password'], self.course_review) self.enroll_student(self.STUDENTS[0]['email'], self.STUDENTS[0]['password'], self.course_review)
self.review_xblock_actual = ItemFactory.create( with self.store.bulk_operations(self.course_actual.id, emit_signals=False):
parent=self.review_unit_actual, review_section_actual = ItemFactory.create(
parent=self.chapter_actual, display_name='Review Subsection'
)
review_unit_actual = ItemFactory.create(
parent=review_section_actual, display_name='Review Unit'
)
review_xblock_actual = ItemFactory.create( # pylint: disable=unused-variable
parent=review_unit_actual,
category='review', category='review',
display_name='Review Tool', display_name='Review Tool'
) )
# Loading the review section # Loading the review section
...@@ -221,25 +226,27 @@ class TestReviewFunctions(TestReviewXBlock): ...@@ -221,25 +226,27 @@ class TestReviewFunctions(TestReviewXBlock):
kwargs={ kwargs={
'course_id': self.course_actual.id, 'course_id': self.course_actual.id,
'chapter': self.chapter_actual.location.name, 'chapter': self.chapter_actual.location.name,
'section': self.review_section_actual.location.name, 'section': review_section_actual.location.name,
} }
)) ))
expected_h2 = 'Nothing to review' expected_h2 = 'Nothing to review'
self.assertIn(expected_h2, response.content) self.assertIn(expected_h2, response.content)
@ddt.data(2, 5, 6, 7) @ddt.data(5, 7)
def test_too_few_review_problems(self, num_desired): def test_too_few_review_problems(self, num_desired):
""" """
If a user does not have enough problems to review, they should If a user does not have enough problems to review, they should
receive a response to go out and try more problems so they have receive a response to go out and try more problems so they have
material to review. material to review.
Testing loading 4 problems and asking for 5 and then loading every
problem and asking for more than that.
""" """
self.enroll_student(self.STUDENTS[0]['email'], self.STUDENTS[0]['password'], self.course_actual) self.enroll_student(self.STUDENTS[0]['email'], self.STUDENTS[0]['password'], self.course_actual)
self.enroll_student(self.STUDENTS[0]['email'], self.STUDENTS[0]['password'], self.course_review) self.enroll_student(self.STUDENTS[0]['email'], self.STUDENTS[0]['password'], self.course_review)
# Want to load fewer problems than num_desired # Want to load fewer problems than num_desired
if num_desired > 4:
self.client.get(reverse( self.client.get(reverse(
'courseware_section', 'courseware_section',
kwargs={ kwargs={
...@@ -257,7 +264,6 @@ class TestReviewFunctions(TestReviewXBlock): ...@@ -257,7 +264,6 @@ class TestReviewFunctions(TestReviewXBlock):
'section': self.section2_actual.location.name, 'section': self.section2_actual.location.name,
} }
)) ))
if num_desired > 6:
self.client.get(reverse( self.client.get(reverse(
'courseware_section', 'courseware_section',
kwargs={ kwargs={
...@@ -267,8 +273,16 @@ class TestReviewFunctions(TestReviewXBlock): ...@@ -267,8 +273,16 @@ class TestReviewFunctions(TestReviewXBlock):
} }
)) ))
self.review_xblock_actual = ItemFactory.create( with self.store.bulk_operations(self.course_actual.id, emit_signals=False):
parent=self.review_unit_actual, review_section_actual = ItemFactory.create(
parent=self.chapter_actual, display_name='Review Subsection'
)
review_unit_actual = ItemFactory.create(
parent=review_section_actual, display_name='Review Unit'
)
review_xblock_actual = ItemFactory.create( # pylint: disable=unused-variable
parent=review_unit_actual,
category='review', category='review',
display_name='Review Tool', display_name='Review Tool',
num_desired=num_desired num_desired=num_desired
...@@ -280,7 +294,7 @@ class TestReviewFunctions(TestReviewXBlock): ...@@ -280,7 +294,7 @@ class TestReviewFunctions(TestReviewXBlock):
kwargs={ kwargs={
'course_id': self.course_actual.id, 'course_id': self.course_actual.id,
'chapter': self.chapter_actual.location.name, 'chapter': self.chapter_actual.location.name,
'section': self.review_section_actual.location.name, 'section': review_section_actual.location.name,
} }
)) ))
...@@ -288,7 +302,7 @@ class TestReviewFunctions(TestReviewXBlock): ...@@ -288,7 +302,7 @@ class TestReviewFunctions(TestReviewXBlock):
self.assertIn(expected_h2, response.content) self.assertIn(expected_h2, response.content)
@ddt.data(2, 3, 4, 5, 6) @ddt.data(2, 6)
def test_review_problems(self, num_desired): def test_review_problems(self, num_desired):
""" """
If a user has enough problems to review, they should If a user has enough problems to review, they should
...@@ -323,19 +337,28 @@ class TestReviewFunctions(TestReviewXBlock): ...@@ -323,19 +337,28 @@ class TestReviewFunctions(TestReviewXBlock):
} }
)) ))
self.review_xblock_actual = ItemFactory.create( with self.store.bulk_operations(self.course_actual.id, emit_signals=False):
parent=self.review_unit_actual, review_section_actual = ItemFactory.create(
parent=self.chapter_actual, display_name='Review Subsection'
)
review_unit_actual = ItemFactory.create(
parent=review_section_actual, display_name='Review Unit'
)
review_xblock_actual = ItemFactory.create( # pylint: disable=unused-variable
parent=review_unit_actual,
category='review', category='review',
display_name='Review Tool', display_name='Review Tool',
num_desired=num_desired num_desired=num_desired
) )
# Loading the review section # Loading the review section
response = self.client.get(reverse( response = self.client.get(reverse(
'courseware_section', 'courseware_section',
kwargs={ kwargs={
'course_id': self.course_actual.id, 'course_id': self.course_actual.id,
'chapter': self.chapter_actual.location.name, 'chapter': self.chapter_actual.location.name,
'section': self.review_section_actual.location.name, 'section': review_section_actual.location.name,
} }
)) ))
...@@ -346,7 +369,8 @@ class TestReviewFunctions(TestReviewXBlock): ...@@ -346,7 +369,8 @@ class TestReviewFunctions(TestReviewXBlock):
expected_correctness_text = 'correct' expected_correctness_text = 'correct'
expected_problems = ['Review Problem 1', 'Review Problem 2', 'Review Problem 3', expected_problems = ['Review Problem 1', 'Review Problem 2', 'Review Problem 3',
'Review Problem 4', 'Review Problem 5', 'Review Problem 6'] 'Review Problem 4', 'Review Problem 5', 'Review Problem 6']
expected_url_beginning = 'https://courses.edx.org/xblock/block-v1:DillonX/DAD101x_review/3T2017+type@problem+block@' expected_url_beginning = settings.LMS_ROOT_URL + \
'/xblock/block-v1:DillonX/DAD101x_review/3T2017+type@problem+block@'
self.assertIn(expected_header_text, response.content) self.assertIn(expected_header_text, response.content)
self.assertEqual(response.content.count(expected_correctness_text), num_desired) self.assertEqual(response.content.count(expected_correctness_text), num_desired)
...@@ -359,7 +383,7 @@ class TestReviewFunctions(TestReviewXBlock): ...@@ -359,7 +383,7 @@ class TestReviewFunctions(TestReviewXBlock):
self.assertEqual(count, num_desired) self.assertEqual(count, num_desired)
self.assertEqual(response.content.count(expected_url_beginning), num_desired) self.assertEqual(response.content.count(expected_url_beginning), num_desired)
@ddt.data(1, 2, 3, 4, 5, 6) @ddt.data(2, 6)
def test_review_problem_urls(self, num_desired): def test_review_problem_urls(self, num_desired):
""" """
Verify that the URLs returned from the Review xBlock are valid and Verify that the URLs returned from the Review xBlock are valid and
...@@ -398,7 +422,7 @@ class TestReviewFunctions(TestReviewXBlock): ...@@ -398,7 +422,7 @@ class TestReviewFunctions(TestReviewXBlock):
crum.set_current_user(user) crum.set_current_user(user)
result_urls = get_review_ids.get_problems(num_desired, self.course_actual.id) result_urls = get_review_ids.get_problems(num_desired, self.course_actual.id)
url_beginning = 'https://courses.edx.org/xblock/block-v1:DillonX/DAD101x_review/3T2017+type@problem+block@' url_beginning = settings.LMS_ROOT_URL + '/xblock/block-v1:DillonX/DAD101x_review/3T2017+type@problem+block@'
expected_urls = [ expected_urls = [
(url_beginning + 'Problem_1', True, 0), (url_beginning + 'Problem_1', True, 0),
(url_beginning + 'Problem_2', True, 0), (url_beginning + 'Problem_2', True, 0),
...@@ -416,13 +440,14 @@ class TestReviewFunctions(TestReviewXBlock): ...@@ -416,13 +440,14 @@ class TestReviewFunctions(TestReviewXBlock):
count += 1 count += 1
self.assertEqual(count, num_desired) self.assertEqual(count, num_desired)
@ddt.data(1, 2, 3, 4, 5) @ddt.data(2, 5)
def test_review_problem_urls_unique_problem(self, num_desired): def test_review_problem_urls_unique_problem(self, num_desired):
""" """
Verify that the URLs returned from the Review xBlock are valid and Verify that the URLs returned from the Review xBlock are valid and
correct URLs for the problems the learner has seen. This test will give correct URLs for the problems the learner has seen. This test will give
a unique problem to a learner and verify only that learner sees a unique problem to a learner and verify only that learner sees
it as a review it as a review. It will also ensure that if a learner has not loaded a
problem, it should never show up as a review problem
""" """
self.enroll_student(self.STUDENTS[0]['email'], self.STUDENTS[0]['password'], self.course_actual) self.enroll_student(self.STUDENTS[0]['email'], self.STUDENTS[0]['password'], self.course_actual)
self.enroll_student(self.STUDENTS[0]['email'], self.STUDENTS[0]['password'], self.course_review) self.enroll_student(self.STUDENTS[0]['email'], self.STUDENTS[0]['password'], self.course_review)
...@@ -449,13 +474,13 @@ class TestReviewFunctions(TestReviewXBlock): ...@@ -449,13 +474,13 @@ class TestReviewFunctions(TestReviewXBlock):
crum.set_current_user(user) crum.set_current_user(user)
result_urls = get_review_ids.get_problems(num_desired, self.course_actual.id) result_urls = get_review_ids.get_problems(num_desired, self.course_actual.id)
url_beginning = 'https://courses.edx.org/xblock/block-v1:DillonX/DAD101x_review/3T2017+type@problem+block@' url_beginning = settings.LMS_ROOT_URL + '/xblock/block-v1:DillonX/DAD101x_review/3T2017+type@problem+block@'
expected_urls = [ expected_urls = [
(url_beginning + 'Problem_1', True, 0), (url_beginning + 'Problem_1', True, 0),
(url_beginning + 'Problem_2', True, 0), (url_beginning + 'Problem_2', True, 0),
(url_beginning + 'Problem_3', True, 0), (url_beginning + 'Problem_3', True, 0),
(url_beginning + 'Problem_4', True, 0), (url_beginning + 'Problem_4', True, 0),
# This is the unique problem # This is the unique problem when num_desired == 5
(url_beginning + 'Problem_6', True, 0) (url_beginning + 'Problem_6', True, 0)
] ]
expected_not_loaded_problem = (url_beginning + 'Problem_5', True, 0) expected_not_loaded_problem = (url_beginning + 'Problem_5', True, 0)
...@@ -469,14 +494,12 @@ class TestReviewFunctions(TestReviewXBlock): ...@@ -469,14 +494,12 @@ class TestReviewFunctions(TestReviewXBlock):
self.assertEqual(count, num_desired) self.assertEqual(count, num_desired)
self.assertNotIn(expected_not_loaded_problem, result_urls) self.assertNotIn(expected_not_loaded_problem, result_urls)
""" # NOTE: This test is failing because when I grab the problem from the CSM,
NOTE: This test is failing because when I grab the problem from the CSM, # it is unable to find its parents. This is some issue with the BlockStructure
it is unable to find its parents. This is some issue with the BlockStructure # and it not being populated the way we want. For now, this is being left out
and it not being populated the way we want. For now, this is being left out # since the first course I'm working with does not use this function.
since the first course I'm working with does not use this function. # TODO: Fix get_vertical from get_review_ids to have the block structure for this test
TODO: Fix get_vertical from get_review_ids to have the block structure for this test # or fix something in this file to make sure it populates the block structure for the CSM
or fix something in this file to make sure it populates the block structure for the CSM
"""
@unittest.skip @unittest.skip
def test_review_vertical_url(self): def test_review_vertical_url(self):
""" """
...@@ -500,7 +523,7 @@ class TestReviewFunctions(TestReviewXBlock): ...@@ -500,7 +523,7 @@ class TestReviewFunctions(TestReviewXBlock):
crum.set_current_user(user) crum.set_current_user(user)
result_url = get_review_ids.get_vertical(self.course_actual.id) result_url = get_review_ids.get_vertical(self.course_actual.id)
expected_url = 'https://courses.edx.org/xblock/block-v1:DillonX/DAD101x_review/3T2017+type@'\ expected_url = settings.LMS_ROOT_URL + \
'vertical+block@New_Unit_1' '/xblock/block-v1:DillonX/DAD101x_review/3T2017+type@vertical+block@New_Unit_1'
self.assertEqual(result_url, expected_url) self.assertEqual(result_url, expected_url)
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment