Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
E
edx-platform
Overview
Overview
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
edx
edx-platform
Commits
e04dbbe1
Commit
e04dbbe1
authored
Feb 17, 2017
by
Eric Fischer
Committed by
GitHub
Feb 17, 2017
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #14529 from edx/efischer/unclean_grades
Unclean Grades
parents
e3fe2f97
7d82f32f
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
27 additions
and
55 deletions
+27
-55
lms/djangoapps/grades/models.py
+1
-22
lms/djangoapps/grades/tests/integration/test_events.py
+5
-3
lms/djangoapps/grades/tests/test_models.py
+6
-19
lms/djangoapps/grades/tests/test_new.py
+1
-1
lms/djangoapps/grades/tests/test_tasks.py
+14
-10
No files found.
lms/djangoapps/grades/models.py
View file @
e04dbbe1
...
@@ -15,7 +15,6 @@ import json
...
@@ -15,7 +15,6 @@ import json
from
lazy
import
lazy
from
lazy
import
lazy
import
logging
import
logging
from
django.core.exceptions
import
ValidationError
from
django.db
import
models
from
django.db
import
models
from
django.utils.timezone
import
now
from
django.utils.timezone
import
now
from
eventtracking
import
tracker
from
eventtracking
import
tracker
...
@@ -295,21 +294,6 @@ class PersistentSubsectionGrade(DeleteGradesMixin, TimeStampedModel):
...
@@ -295,21 +294,6 @@ class PersistentSubsectionGrade(DeleteGradesMixin, TimeStampedModel):
# track which blocks were visible at the time of grade calculation
# track which blocks were visible at the time of grade calculation
visible_blocks
=
models
.
ForeignKey
(
VisibleBlocks
,
db_column
=
'visible_blocks_hash'
,
to_field
=
'hashed'
)
visible_blocks
=
models
.
ForeignKey
(
VisibleBlocks
,
db_column
=
'visible_blocks_hash'
,
to_field
=
'hashed'
)
def
_is_unattempted_with_score
(
self
):
"""
Return True if the object has a non-zero score, but has not been
attempted. This is an inconsistent state, and needs to be cleaned up.
"""
return
self
.
first_attempted
is
None
and
any
(
field
!=
0.0
for
field
in
(
self
.
earned_all
,
self
.
earned_graded
))
def
clean
(
self
):
"""
If an grade has not been attempted, but was given a non-zero score,
raise a ValidationError.
"""
if
self
.
_is_unattempted_with_score
():
raise
ValidationError
(
"Unattempted problems cannot have a non-zero score."
)
@property
@property
def
full_usage_key
(
self
):
def
full_usage_key
(
self
):
"""
"""
...
@@ -391,7 +375,6 @@ class PersistentSubsectionGrade(DeleteGradesMixin, TimeStampedModel):
...
@@ -391,7 +375,6 @@ class PersistentSubsectionGrade(DeleteGradesMixin, TimeStampedModel):
if
attempted
and
not
grade
.
first_attempted
:
if
attempted
and
not
grade
.
first_attempted
:
grade
.
first_attempted
=
now
()
grade
.
first_attempted
=
now
()
grade
.
save
()
grade
.
save
()
grade
.
full_clean
()
cls
.
_emit_grade_calculated_event
(
grade
)
cls
.
_emit_grade_calculated_event
(
grade
)
return
grade
return
grade
...
@@ -402,9 +385,7 @@ class PersistentSubsectionGrade(DeleteGradesMixin, TimeStampedModel):
...
@@ -402,9 +385,7 @@ class PersistentSubsectionGrade(DeleteGradesMixin, TimeStampedModel):
"""
"""
cls
.
_prepare_params_and_visible_blocks
(
params
)
cls
.
_prepare_params_and_visible_blocks
(
params
)
cls
.
_prepare_attempted_for_create
(
params
,
now
())
cls
.
_prepare_attempted_for_create
(
params
,
now
())
grade
=
cls
(
**
params
)
grade
=
cls
.
objects
.
create
(
**
params
)
grade
.
full_clean
()
grade
.
save
()
cls
.
_emit_grade_calculated_event
(
grade
)
cls
.
_emit_grade_calculated_event
(
grade
)
return
grade
return
grade
...
@@ -423,8 +404,6 @@ class PersistentSubsectionGrade(DeleteGradesMixin, TimeStampedModel):
...
@@ -423,8 +404,6 @@ class PersistentSubsectionGrade(DeleteGradesMixin, TimeStampedModel):
for
params
in
grade_params_iter
:
for
params
in
grade_params_iter
:
cls
.
_prepare_attempted_for_create
(
params
,
first_attempt_timestamp
)
cls
.
_prepare_attempted_for_create
(
params
,
first_attempt_timestamp
)
grades
=
[
PersistentSubsectionGrade
(
**
params
)
for
params
in
grade_params_iter
]
grades
=
[
PersistentSubsectionGrade
(
**
params
)
for
params
in
grade_params_iter
]
for
grade
in
grades
:
grade
.
full_clean
()
grades
=
cls
.
objects
.
bulk_create
(
grades
)
grades
=
cls
.
objects
.
bulk_create
(
grades
)
for
grade
in
grades
:
for
grade
in
grades
:
cls
.
_emit_grade_calculated_event
(
grade
)
cls
.
_emit_grade_calculated_event
(
grade
)
...
...
lms/djangoapps/grades/tests/integration/test_events.py
View file @
e04dbbe1
...
@@ -8,7 +8,6 @@ from lms.djangoapps.instructor.enrollment import reset_student_attempts
...
@@ -8,7 +8,6 @@ from lms.djangoapps.instructor.enrollment import reset_student_attempts
from
lms.djangoapps.instructor_task.api
import
submit_rescore_problem_for_student
from
lms.djangoapps.instructor_task.api
import
submit_rescore_problem_for_student
from
mock
import
patch
from
mock
import
patch
from
openedx.core.djangolib.testing.utils
import
get_mock_request
from
openedx.core.djangolib.testing.utils
import
get_mock_request
from
xmodule.modulestore.django
import
modulestore
from
xmodule.modulestore.tests.django_utils
import
SharedModuleStoreTestCase
from
xmodule.modulestore.tests.django_utils
import
SharedModuleStoreTestCase
from
xmodule.modulestore.tests.factories
import
CourseFactory
,
ItemFactory
from
xmodule.modulestore.tests.factories
import
CourseFactory
,
ItemFactory
...
@@ -28,8 +27,10 @@ class GradesEventIntegrationTest(ProblemSubmissionTestMixin, SharedModuleStoreTe
...
@@ -28,8 +27,10 @@ class GradesEventIntegrationTest(ProblemSubmissionTestMixin, SharedModuleStoreTe
of the grading infrastructure.
of the grading infrastructure.
"""
"""
@classmethod
@classmethod
def
setUpClass
(
cls
):
def
reset_course
(
cls
):
super
(
GradesEventIntegrationTest
,
cls
)
.
setUpClass
()
"""
Sets up the course anew.
"""
with
cls
.
store
.
default_store
(
ModuleStoreEnum
.
Type
.
split
):
with
cls
.
store
.
default_store
(
ModuleStoreEnum
.
Type
.
split
):
cls
.
course
=
CourseFactory
.
create
()
cls
.
course
=
CourseFactory
.
create
()
cls
.
chapter
=
ItemFactory
.
create
(
cls
.
chapter
=
ItemFactory
.
create
(
...
@@ -63,6 +64,7 @@ class GradesEventIntegrationTest(ProblemSubmissionTestMixin, SharedModuleStoreTe
...
@@ -63,6 +64,7 @@ class GradesEventIntegrationTest(ProblemSubmissionTestMixin, SharedModuleStoreTe
)
)
def
setUp
(
self
):
def
setUp
(
self
):
self
.
reset_course
()
super
(
GradesEventIntegrationTest
,
self
)
.
setUp
()
super
(
GradesEventIntegrationTest
,
self
)
.
setUp
()
self
.
request
=
get_mock_request
(
UserFactory
())
self
.
request
=
get_mock_request
(
UserFactory
())
self
.
student
=
self
.
request
.
user
self
.
student
=
self
.
request
.
user
...
...
lms/djangoapps/grades/tests/test_models.py
View file @
e04dbbe1
...
@@ -9,7 +9,6 @@ from hashlib import sha1
...
@@ -9,7 +9,6 @@ from hashlib import sha1
import
json
import
json
from
mock
import
patch
from
mock
import
patch
from
django.core.exceptions
import
ValidationError
from
django.db.utils
import
IntegrityError
from
django.db.utils
import
IntegrityError
from
django.test
import
TestCase
from
django.test
import
TestCase
from
django.utils.timezone
import
now
from
django.utils.timezone
import
now
...
@@ -228,7 +227,7 @@ class PersistentSubsectionGradeTest(GradesModelTestCase):
...
@@ -228,7 +227,7 @@ class PersistentSubsectionGradeTest(GradesModelTestCase):
)
)
self
.
assertEqual
(
created_grade
,
read_grade
)
self
.
assertEqual
(
created_grade
,
read_grade
)
self
.
assertEqual
(
read_grade
.
visible_blocks
.
blocks
,
self
.
block_records
)
self
.
assertEqual
(
read_grade
.
visible_blocks
.
blocks
,
self
.
block_records
)
with
self
.
assertRaises
(
Validation
Error
):
with
self
.
assertRaises
(
Integrity
Error
):
PersistentSubsectionGrade
.
create_grade
(
**
self
.
params
)
PersistentSubsectionGrade
.
create_grade
(
**
self
.
params
)
@ddt.data
(
'course_version'
,
'subtree_edited_timestamp'
)
@ddt.data
(
'course_version'
,
'subtree_edited_timestamp'
)
...
@@ -237,12 +236,12 @@ class PersistentSubsectionGradeTest(GradesModelTestCase):
...
@@ -237,12 +236,12 @@ class PersistentSubsectionGradeTest(GradesModelTestCase):
PersistentSubsectionGrade
.
create_grade
(
**
self
.
params
)
PersistentSubsectionGrade
.
create_grade
(
**
self
.
params
)
@ddt.data
(
@ddt.data
(
(
"user_id"
,
Validation
Error
),
(
"user_id"
,
Integrity
Error
),
(
"usage_key"
,
KeyError
),
(
"usage_key"
,
KeyError
),
(
"earned_all"
,
Validation
Error
),
(
"earned_all"
,
Integrity
Error
),
(
"possible_all"
,
Validation
Error
),
(
"possible_all"
,
Integrity
Error
),
(
"earned_graded"
,
Validation
Error
),
(
"earned_graded"
,
Integrity
Error
),
(
"possible_graded"
,
Validation
Error
),
(
"possible_graded"
,
Integrity
Error
),
(
"visible_blocks"
,
KeyError
),
(
"visible_blocks"
,
KeyError
),
(
"attempted"
,
KeyError
),
(
"attempted"
,
KeyError
),
)
)
...
@@ -276,18 +275,6 @@ class PersistentSubsectionGradeTest(GradesModelTestCase):
...
@@ -276,18 +275,6 @@ class PersistentSubsectionGradeTest(GradesModelTestCase):
self
.
assertEqual
(
grade
.
earned_all
,
0.0
)
self
.
assertEqual
(
grade
.
earned_all
,
0.0
)
self
.
assertEqual
(
grade
.
earned_graded
,
0.0
)
self
.
assertEqual
(
grade
.
earned_graded
,
0.0
)
def
test_create_inconsistent_unattempted
(
self
):
self
.
params
[
'attempted'
]
=
False
with
self
.
assertRaises
(
ValidationError
):
PersistentSubsectionGrade
.
create_grade
(
**
self
.
params
)
def
test_update_or_create_inconsistent_unattempted
(
self
):
self
.
params
[
'attempted'
]
=
False
self
.
params
[
'earned_all'
]
=
1.0
self
.
params
[
'earned_graded'
]
=
1.0
with
self
.
assertRaises
(
ValidationError
):
PersistentSubsectionGrade
.
update_or_create_grade
(
**
self
.
params
)
def
test_first_attempted_not_changed_on_update
(
self
):
def
test_first_attempted_not_changed_on_update
(
self
):
PersistentSubsectionGrade
.
create_grade
(
**
self
.
params
)
PersistentSubsectionGrade
.
create_grade
(
**
self
.
params
)
moment
=
now
()
moment
=
now
()
...
...
lms/djangoapps/grades/tests/test_new.py
View file @
e04dbbe1
...
@@ -214,7 +214,7 @@ class TestSubsectionGradeFactory(ProblemSubmissionTestMixin, GradeTestBase):
...
@@ -214,7 +214,7 @@ class TestSubsectionGradeFactory(ProblemSubmissionTestMixin, GradeTestBase):
'lms.djangoapps.grades.new.subsection_grade.SubsectionGradeFactory._get_bulk_cached_grade'
,
'lms.djangoapps.grades.new.subsection_grade.SubsectionGradeFactory._get_bulk_cached_grade'
,
wraps
=
self
.
subsection_grade_factory
.
_get_bulk_cached_grade
wraps
=
self
.
subsection_grade_factory
.
_get_bulk_cached_grade
)
as
mock_get_bulk_cached_grade
:
)
as
mock_get_bulk_cached_grade
:
with
self
.
assertNumQueries
(
1
4
):
with
self
.
assertNumQueries
(
1
2
):
grade_a
=
self
.
subsection_grade_factory
.
create
(
self
.
sequence
)
grade_a
=
self
.
subsection_grade_factory
.
create
(
self
.
sequence
)
self
.
assertTrue
(
mock_get_bulk_cached_grade
.
called
)
self
.
assertTrue
(
mock_get_bulk_cached_grade
.
called
)
self
.
assertTrue
(
mock_create_grade
.
called
)
self
.
assertTrue
(
mock_create_grade
.
called
)
...
...
lms/djangoapps/grades/tests/test_tasks.py
View file @
e04dbbe1
...
@@ -42,7 +42,7 @@ class RecalculateSubsectionGradeTest(ModuleStoreTestCase):
...
@@ -42,7 +42,7 @@ class RecalculateSubsectionGradeTest(ModuleStoreTestCase):
self
.
user
=
UserFactory
()
self
.
user
=
UserFactory
()
PersistentGradesEnabledFlag
.
objects
.
create
(
enabled_for_all_courses
=
True
,
enabled
=
True
)
PersistentGradesEnabledFlag
.
objects
.
create
(
enabled_for_all_courses
=
True
,
enabled
=
True
)
def
set_up_course
(
self
,
enable_persistent_grades
=
True
):
def
set_up_course
(
self
,
enable_persistent_grades
=
True
,
create_multiple_subsections
=
False
):
"""
"""
Configures the course for this test.
Configures the course for this test.
"""
"""
...
@@ -59,6 +59,10 @@ class RecalculateSubsectionGradeTest(ModuleStoreTestCase):
...
@@ -59,6 +59,10 @@ class RecalculateSubsectionGradeTest(ModuleStoreTestCase):
self
.
sequential
=
ItemFactory
.
create
(
parent
=
self
.
chapter
,
category
=
'sequential'
,
display_name
=
"Sequential1"
)
self
.
sequential
=
ItemFactory
.
create
(
parent
=
self
.
chapter
,
category
=
'sequential'
,
display_name
=
"Sequential1"
)
self
.
problem
=
ItemFactory
.
create
(
parent
=
self
.
sequential
,
category
=
'problem'
,
display_name
=
'Problem'
)
self
.
problem
=
ItemFactory
.
create
(
parent
=
self
.
sequential
,
category
=
'problem'
,
display_name
=
'Problem'
)
if
create_multiple_subsections
:
seq2
=
ItemFactory
.
create
(
parent
=
self
.
chapter
,
category
=
'sequential'
)
ItemFactory
.
create
(
parent
=
seq2
,
category
=
'problem'
)
self
.
frozen_now_datetime
=
datetime
.
now
()
.
replace
(
tzinfo
=
pytz
.
UTC
)
self
.
frozen_now_datetime
=
datetime
.
now
()
.
replace
(
tzinfo
=
pytz
.
UTC
)
self
.
frozen_now_timestamp
=
to_timestamp
(
self
.
frozen_now_datetime
)
self
.
frozen_now_timestamp
=
to_timestamp
(
self
.
frozen_now_datetime
)
...
@@ -137,28 +141,28 @@ class RecalculateSubsectionGradeTest(ModuleStoreTestCase):
...
@@ -137,28 +141,28 @@ class RecalculateSubsectionGradeTest(ModuleStoreTestCase):
self
.
assertEquals
(
mock_block_structure_create
.
call_count
,
1
)
self
.
assertEquals
(
mock_block_structure_create
.
call_count
,
1
)
@ddt.data
(
@ddt.data
(
(
ModuleStoreEnum
.
Type
.
mongo
,
1
,
23
),
(
ModuleStoreEnum
.
Type
.
mongo
,
1
,
24
,
True
),
(
ModuleStoreEnum
.
Type
.
split
,
3
,
22
),
(
ModuleStoreEnum
.
Type
.
mongo
,
1
,
21
,
False
),
(
ModuleStoreEnum
.
Type
.
split
,
3
,
23
,
True
),
(
ModuleStoreEnum
.
Type
.
split
,
3
,
20
,
False
),
)
)
@ddt.unpack
@ddt.unpack
def
test_query_counts
(
self
,
default_store
,
num_mongo_calls
,
num_sql_calls
):
def
test_query_counts
(
self
,
default_store
,
num_mongo_calls
,
num_sql_calls
,
create_multiple_subsections
):
with
self
.
store
.
default_store
(
default_store
):
with
self
.
store
.
default_store
(
default_store
):
self
.
set_up_course
()
self
.
set_up_course
(
create_multiple_subsections
=
create_multiple_subsections
)
self
.
assertTrue
(
PersistentGradesEnabledFlag
.
feature_enabled
(
self
.
course
.
id
))
self
.
assertTrue
(
PersistentGradesEnabledFlag
.
feature_enabled
(
self
.
course
.
id
))
with
check_mongo_calls
(
num_mongo_calls
):
with
check_mongo_calls
(
num_mongo_calls
):
with
self
.
assertNumQueries
(
num_sql_calls
):
with
self
.
assertNumQueries
(
num_sql_calls
):
self
.
_apply_recalculate_subsection_grade
()
self
.
_apply_recalculate_subsection_grade
()
# TODO (TNL-6225) Fix the number of SQL queries so they
# don't grow linearly with the number of sequentials.
@ddt.data
(
@ddt.data
(
(
ModuleStoreEnum
.
Type
.
mongo
,
1
,
46
),
(
ModuleStoreEnum
.
Type
.
mongo
,
1
,
24
),
(
ModuleStoreEnum
.
Type
.
split
,
3
,
45
),
(
ModuleStoreEnum
.
Type
.
split
,
3
,
23
),
)
)
@ddt.unpack
@ddt.unpack
def
test_query_counts_dont_change_with_more_content
(
self
,
default_store
,
num_mongo_calls
,
num_sql_calls
):
def
test_query_counts_dont_change_with_more_content
(
self
,
default_store
,
num_mongo_calls
,
num_sql_calls
):
with
self
.
store
.
default_store
(
default_store
):
with
self
.
store
.
default_store
(
default_store
):
self
.
set_up_course
()
self
.
set_up_course
(
create_multiple_subsections
=
True
)
self
.
assertTrue
(
PersistentGradesEnabledFlag
.
feature_enabled
(
self
.
course
.
id
))
self
.
assertTrue
(
PersistentGradesEnabledFlag
.
feature_enabled
(
self
.
course
.
id
))
num_problems
=
10
num_problems
=
10
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment