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
ba00a442
Commit
ba00a442
authored
Mar 25, 2014
by
Will Daly
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #3037 from edx/will/tim-installation-rebase
Install new ORA (edx-tim)
parents
274224bf
440d9ddb
Hide whitespace changes
Inline
Side-by-side
Showing
7 changed files
with
92 additions
and
7 deletions
+92
-7
cms/djangoapps/contentstore/tests/test_contentstore.py
+1
-1
cms/djangoapps/contentstore/views/component.py
+1
-0
cms/envs/common.py
+7
-0
lms/djangoapps/courseware/grades.py
+36
-5
lms/djangoapps/courseware/tests/test_submitting_problems.py
+39
-1
lms/envs/common.py
+7
-0
requirements/edx/github.txt
+1
-0
No files found.
cms/djangoapps/contentstore/tests/test_contentstore.py
View file @
ba00a442
...
@@ -143,7 +143,7 @@ class ContentStoreToyCourseTest(ModuleStoreTestCase):
...
@@ -143,7 +143,7 @@ class ContentStoreToyCourseTest(ModuleStoreTestCase):
self
.
check_components_on_page
(
self
.
check_components_on_page
(
ADVANCED_COMPONENT_TYPES
,
ADVANCED_COMPONENT_TYPES
,
[
'Word cloud'
,
'Annotation'
,
'Text Annotation'
,
'Video Annotation'
,
[
'Word cloud'
,
'Annotation'
,
'Text Annotation'
,
'Video Annotation'
,
'Open Response Assessment'
,
'Peer Grading Interface'
],
'Open Response Assessment'
,
'Peer Grading Interface'
,
'openassessment'
],
)
)
def
test_advanced_components_require_two_clicks
(
self
):
def
test_advanced_components_require_two_clicks
(
self
):
...
...
cms/djangoapps/contentstore/views/component.py
View file @
ba00a442
...
@@ -60,6 +60,7 @@ else:
...
@@ -60,6 +60,7 @@ else:
'word_cloud'
,
'word_cloud'
,
'graphical_slider_tool'
,
'graphical_slider_tool'
,
'lti'
,
'lti'
,
'openassessment'
,
# edx-ora2
]
+
OPEN_ENDED_COMPONENT_TYPES
+
NOTE_COMPONENT_TYPES
]
+
OPEN_ENDED_COMPONENT_TYPES
+
NOTE_COMPONENT_TYPES
ADVANCED_COMPONENT_CATEGORY
=
'advanced'
ADVANCED_COMPONENT_CATEGORY
=
'advanced'
...
...
cms/envs/common.py
View file @
ba00a442
...
@@ -553,6 +553,13 @@ MAX_FAILED_LOGIN_ATTEMPTS_LOCKOUT_PERIOD_SECS = 15 * 60
...
@@ -553,6 +553,13 @@ MAX_FAILED_LOGIN_ATTEMPTS_LOCKOUT_PERIOD_SECS = 15 * 60
OPTIONAL_APPS
=
(
OPTIONAL_APPS
=
(
'edx_jsdraw'
,
'edx_jsdraw'
,
'mentoring'
,
'mentoring'
,
# edx-ora2
'submissions'
,
'openassessment'
,
'openassessment.assessment'
,
'openassessment.workflow'
,
'openassessment.xblock'
)
)
for
app_name
in
OPTIONAL_APPS
:
for
app_name
in
OPTIONAL_APPS
:
...
...
lms/djangoapps/courseware/grades.py
View file @
ba00a442
...
@@ -14,6 +14,8 @@ from dogapi import dog_stats_api
...
@@ -14,6 +14,8 @@ from dogapi import dog_stats_api
from
courseware
import
courses
from
courseware
import
courses
from
courseware.model_data
import
FieldDataCache
from
courseware.model_data
import
FieldDataCache
from
student.models
import
anonymous_id_for_user
from
submissions
import
api
as
sub_api
from
xmodule
import
graders
from
xmodule
import
graders
from
xmodule.graders
import
Score
from
xmodule.graders
import
Score
from
xmodule.modulestore.django
import
modulestore
from
xmodule.modulestore.django
import
modulestore
...
@@ -178,6 +180,11 @@ def _grade(student, request, course, keep_raw_scores):
...
@@ -178,6 +180,11 @@ def _grade(student, request, course, keep_raw_scores):
grading_context
=
course
.
grading_context
grading_context
=
course
.
grading_context
raw_scores
=
[]
raw_scores
=
[]
# Dict of item_ids -> (earned, possible) point tuples. This *only* grabs
# scores that were registered with the submissions API, which for the moment
# means only openassessment (edx-ora2)
submissions_scores
=
sub_api
.
get_scores
(
course
.
id
,
anonymous_id_for_user
(
student
,
course
.
id
))
totaled_scores
=
{}
totaled_scores
=
{}
# This next complicated loop is just to collect the totaled_scores, which is
# This next complicated loop is just to collect the totaled_scores, which is
# passed to the grader
# passed to the grader
...
@@ -194,7 +201,15 @@ def _grade(student, request, course, keep_raw_scores):
...
@@ -194,7 +201,15 @@ def _grade(student, request, course, keep_raw_scores):
descriptor
.
always_recalculate_grades
for
descriptor
in
section
[
'xmoduledescriptors'
]
descriptor
.
always_recalculate_grades
for
descriptor
in
section
[
'xmoduledescriptors'
]
)
)
# If we haven't seen a single problem in the section, we don't have to grade it at all! We can assume 0%
# If there are no problems that always have to be regraded, check to
# see if any of our locations are in the scores from the submissions
# API. If scores exist, we have to calculate grades for this section.
if
not
should_grade_section
:
should_grade_section
=
any
(
descriptor
.
location
.
url
()
in
submissions_scores
for
descriptor
in
section
[
'xmoduledescriptors'
]
)
if
not
should_grade_section
:
if
not
should_grade_section
:
with
manual_transaction
():
with
manual_transaction
():
should_grade_section
=
StudentModule
.
objects
.
filter
(
should_grade_section
=
StudentModule
.
objects
.
filter
(
...
@@ -204,6 +219,8 @@ def _grade(student, request, course, keep_raw_scores):
...
@@ -204,6 +219,8 @@ def _grade(student, request, course, keep_raw_scores):
]
]
)
.
exists
()
)
.
exists
()
# If we haven't seen a single problem in the section, we don't have
# to grade it at all! We can assume 0%
if
should_grade_section
:
if
should_grade_section
:
scores
=
[]
scores
=
[]
...
@@ -217,7 +234,9 @@ def _grade(student, request, course, keep_raw_scores):
...
@@ -217,7 +234,9 @@ def _grade(student, request, course, keep_raw_scores):
for
module_descriptor
in
yield_dynamic_descriptor_descendents
(
section_descriptor
,
create_module
):
for
module_descriptor
in
yield_dynamic_descriptor_descendents
(
section_descriptor
,
create_module
):
(
correct
,
total
)
=
get_score
(
course
.
id
,
student
,
module_descriptor
,
create_module
)
(
correct
,
total
)
=
get_score
(
course
.
id
,
student
,
module_descriptor
,
create_module
,
scores_cache
=
submissions_scores
)
if
correct
is
None
and
total
is
None
:
if
correct
is
None
and
total
is
None
:
continue
continue
...
@@ -331,6 +350,8 @@ def _progress_summary(student, request, course):
...
@@ -331,6 +350,8 @@ def _progress_summary(student, request, course):
# This student must not have access to the course.
# This student must not have access to the course.
return
None
return
None
submissions_scores
=
sub_api
.
get_scores
(
course
.
id
,
anonymous_id_for_user
(
student
,
course
.
id
))
chapters
=
[]
chapters
=
[]
# Don't include chapters that aren't displayable (e.g. due to error)
# Don't include chapters that aren't displayable (e.g. due to error)
for
chapter_module
in
course_module
.
get_display_items
():
for
chapter_module
in
course_module
.
get_display_items
():
...
@@ -353,7 +374,9 @@ def _progress_summary(student, request, course):
...
@@ -353,7 +374,9 @@ def _progress_summary(student, request, course):
for
module_descriptor
in
yield_dynamic_descriptor_descendents
(
section_module
,
module_creator
):
for
module_descriptor
in
yield_dynamic_descriptor_descendents
(
section_module
,
module_creator
):
course_id
=
course
.
id
course_id
=
course
.
id
(
correct
,
total
)
=
get_score
(
course_id
,
student
,
module_descriptor
,
module_creator
)
(
correct
,
total
)
=
get_score
(
course_id
,
student
,
module_descriptor
,
module_creator
,
scores_cache
=
submissions_scores
)
if
correct
is
None
and
total
is
None
:
if
correct
is
None
and
total
is
None
:
continue
continue
...
@@ -383,7 +406,8 @@ def _progress_summary(student, request, course):
...
@@ -383,7 +406,8 @@ def _progress_summary(student, request, course):
return
chapters
return
chapters
def
get_score
(
course_id
,
user
,
problem_descriptor
,
module_creator
):
def
get_score
(
course_id
,
user
,
problem_descriptor
,
module_creator
,
scores_cache
=
None
):
"""
"""
Return the score for a user on a problem, as a tuple (correct, total).
Return the score for a user on a problem, as a tuple (correct, total).
e.g. (5,7) if you got 5 out of 7 points.
e.g. (5,7) if you got 5 out of 7 points.
...
@@ -395,11 +419,18 @@ def get_score(course_id, user, problem_descriptor, module_creator):
...
@@ -395,11 +419,18 @@ def get_score(course_id, user, problem_descriptor, module_creator):
problem_descriptor: an XModuleDescriptor
problem_descriptor: an XModuleDescriptor
module_creator: a function that takes a descriptor, and returns the corresponding XModule for this user.
module_creator: a function that takes a descriptor, and returns the corresponding XModule for this user.
Can return None if user doesn't have access, or if something else went wrong.
Can return None if user doesn't have access, or if something else went wrong.
cache: A FieldDataCache
scores_cache: A dict of location names to (earned, possible) point tuples.
If an entry is found in this cache, it takes precedence.
"""
"""
scores_cache
=
scores_cache
or
{}
if
not
user
.
is_authenticated
():
if
not
user
.
is_authenticated
():
return
(
None
,
None
)
return
(
None
,
None
)
location_url
=
problem_descriptor
.
location
.
url
()
if
location_url
in
scores_cache
:
return
scores_cache
[
location_url
]
# some problems have state that is updated independently of interaction
# some problems have state that is updated independently of interaction
# with the LMS, so they need to always be scored. (E.g. foldit.)
# with the LMS, so they need to always be scored. (E.g. foldit.)
if
problem_descriptor
.
always_recalculate_grades
:
if
problem_descriptor
.
always_recalculate_grades
:
...
...
lms/djangoapps/courseware/tests/test_submitting_problems.py
View file @
ba00a442
...
@@ -467,7 +467,7 @@ class TestCourseGrader(TestSubmittingProblems):
...
@@ -467,7 +467,7 @@ class TestCourseGrader(TestSubmittingProblems):
self
.
check_grade_percent
(
1.0
)
self
.
check_grade_percent
(
1.0
)
self
.
assertEqual
(
self
.
get_grade_summary
()[
'grade'
],
'A'
)
self
.
assertEqual
(
self
.
get_grade_summary
()[
'grade'
],
'A'
)
def
test_wrong_a
sn
wers
(
self
):
def
test_wrong_a
ns
wers
(
self
):
"""
"""
Check that answering incorrectly is graded properly.
Check that answering incorrectly is graded properly.
"""
"""
...
@@ -478,6 +478,44 @@ class TestCourseGrader(TestSubmittingProblems):
...
@@ -478,6 +478,44 @@ class TestCourseGrader(TestSubmittingProblems):
self
.
check_grade_percent
(
0.67
)
self
.
check_grade_percent
(
0.67
)
self
.
assertEqual
(
self
.
get_grade_summary
()[
'grade'
],
'B'
)
self
.
assertEqual
(
self
.
get_grade_summary
()[
'grade'
],
'B'
)
def
test_submissions_api_overrides_scores
(
self
):
"""
Check that answering incorrectly is graded properly.
"""
self
.
basic_setup
()
self
.
submit_question_answer
(
'p1'
,
{
'2_1'
:
'Correct'
})
self
.
submit_question_answer
(
'p2'
,
{
'2_1'
:
'Correct'
})
self
.
submit_question_answer
(
'p3'
,
{
'2_1'
:
'Incorrect'
})
self
.
check_grade_percent
(
0.67
)
self
.
assertEqual
(
self
.
get_grade_summary
()[
'grade'
],
'B'
)
# But now we mock out a get_scores call, and watch as it overrides the
# score read from StudentModule and our student gets an A instead.
with
patch
(
'submissions.api.get_scores'
)
as
mock_get_scores
:
mock_get_scores
.
return_value
=
{
self
.
problem_location
(
'p3'
):
(
1
,
1
)
}
self
.
check_grade_percent
(
1.0
)
self
.
assertEqual
(
self
.
get_grade_summary
()[
'grade'
],
'A'
)
def
test_submissions_api_anonymous_student_id
(
self
):
"""
Check that the submissions API is sent an anonymous student ID.
"""
self
.
basic_setup
()
self
.
submit_question_answer
(
'p1'
,
{
'2_1'
:
'Correct'
})
self
.
submit_question_answer
(
'p2'
,
{
'2_1'
:
'Correct'
})
self
.
submit_question_answer
(
'p3'
,
{
'2_1'
:
'Incorrect'
})
with
patch
(
'submissions.api.get_scores'
)
as
mock_get_scores
:
mock_get_scores
.
return_value
=
{
self
.
problem_location
(
'p3'
):
(
1
,
1
)
}
self
.
get_grade_summary
()
# Verify that the submissions API was sent an anonymized student ID
mock_get_scores
.
assert_called_with
(
self
.
course
.
id
,
'99ac6730dc5f900d69fd735975243b31'
)
def
test_weighted_homework
(
self
):
def
test_weighted_homework
(
self
):
"""
"""
Test that the homework section has proper weight.
Test that the homework section has proper weight.
...
...
lms/envs/common.py
View file @
ba00a442
...
@@ -1489,6 +1489,13 @@ ALL_LANGUAGES = (
...
@@ -1489,6 +1489,13 @@ ALL_LANGUAGES = (
OPTIONAL_APPS
=
(
OPTIONAL_APPS
=
(
'edx_jsdraw'
,
'edx_jsdraw'
,
'mentoring'
,
'mentoring'
,
# edx-ora2
'submissions'
,
'openassessment'
,
'openassessment.assessment'
,
'openassessment.workflow'
,
'openassessment.xblock'
)
)
for
app_name
in
OPTIONAL_APPS
:
for
app_name
in
OPTIONAL_APPS
:
...
...
requirements/edx/github.txt
View file @
ba00a442
...
@@ -26,3 +26,4 @@
...
@@ -26,3 +26,4 @@
-e git+https://github.com/edx/bok-choy.git@25a47b3bf87c503fc4996e52addac83b42ec6f38#egg=bok_choy
-e git+https://github.com/edx/bok-choy.git@25a47b3bf87c503fc4996e52addac83b42ec6f38#egg=bok_choy
-e git+https://github.com/edx-solutions/django-splash.git@9965a53c269666a30bb4e2b3f6037c138aef2a55#egg=django-splash
-e git+https://github.com/edx-solutions/django-splash.git@9965a53c269666a30bb4e2b3f6037c138aef2a55#egg=django-splash
-e git+https://github.com/edx/acid-block.git@459aff7b63db8f2c5decd1755706c1a64fb4ebb1#egg=acid-xblock
-e git+https://github.com/edx/acid-block.git@459aff7b63db8f2c5decd1755706c1a64fb4ebb1#egg=acid-xblock
-e git+https://github.com/edx/edx-ora2.git@87fd72f9927cd37e553d7f68cfaf80d6c574eb55#egg=edx-ora2
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