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
9269ec3b
Commit
9269ec3b
authored
Apr 22, 2015
by
Daniel Friedman
Committed by
Diana Huang
May 12, 2015
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Add new instructor task for weighted problems
parent
9c32b1e8
Hide whitespace changes
Inline
Side-by-side
Showing
17 changed files
with
433 additions
and
18 deletions
+433
-18
lms/djangoapps/course_structure_api/v0/views.py
+0
-1
lms/djangoapps/courseware/grades.py
+3
-3
lms/djangoapps/courseware/tests/test_grades.py
+3
-3
lms/djangoapps/django_comment_client/tests/utils.py
+9
-0
lms/djangoapps/instructor/views/api.py
+28
-0
lms/djangoapps/instructor/views/api_urls.py
+2
-0
lms/djangoapps/instructor/views/instructor_dashboard.py
+1
-0
lms/djangoapps/instructor_task/api.py
+13
-0
lms/djangoapps/instructor_task/tasks.py
+20
-0
lms/djangoapps/instructor_task/tasks_helper.py
+103
-0
lms/djangoapps/instructor_task/tests/test_base.py
+15
-8
lms/djangoapps/instructor_task/tests/test_integration.py
+1
-1
lms/djangoapps/instructor_task/tests/test_tasks_helper.py
+166
-0
lms/static/coffee/src/instructor_dashboard/data_download.coffee
+9
-2
lms/templates/instructor/instructor_dashboard_2/data_download.html
+2
-0
openedx/core/djangoapps/content/course_structures/models.py
+24
-0
openedx/core/djangoapps/content/course_structures/tests.py
+34
-0
No files found.
lms/djangoapps/course_structure_api/v0/views.py
View file @
9269ec3b
...
@@ -262,7 +262,6 @@ class CourseStructure(CourseViewMixin, RetrieveAPIView):
...
@@ -262,7 +262,6 @@ class CourseStructure(CourseViewMixin, RetrieveAPIView):
return
Response
(
status
=
503
,
headers
=
{
'Retry-After'
:
'120'
})
return
Response
(
status
=
503
,
headers
=
{
'Retry-After'
:
'120'
})
class
CourseGradingPolicy
(
CourseViewMixin
,
ListAPIView
):
class
CourseGradingPolicy
(
CourseViewMixin
,
ListAPIView
):
"""
"""
**Use Case**
**Use Case**
...
...
lms/djangoapps/courseware/grades.py
View file @
9269ec3b
...
@@ -225,7 +225,7 @@ def _grade(student, request, course, keep_raw_scores):
...
@@ -225,7 +225,7 @@ def _grade(student, request, course, keep_raw_scores):
graded
=
module_descriptor
.
graded
graded
=
module_descriptor
.
graded
if
not
total
>
0
:
if
not
total
>
0
:
#We simply cannot grade a problem that is 12/0, because we might need it as a percentage
#
We simply cannot grade a problem that is 12/0, because we might need it as a percentage
graded
=
False
graded
=
False
scores
.
append
(
scores
.
append
(
...
@@ -494,7 +494,7 @@ def manual_transaction():
...
@@ -494,7 +494,7 @@ def manual_transaction():
transaction
.
commit
()
transaction
.
commit
()
def
iterate_grades_for
(
course_or_id
,
students
):
def
iterate_grades_for
(
course_or_id
,
students
,
keep_raw_scores
=
False
):
"""Given a course_id and an iterable of students (User), yield a tuple of:
"""Given a course_id and an iterable of students (User), yield a tuple of:
(student, gradeset, err_msg) for every student enrolled in the course.
(student, gradeset, err_msg) for every student enrolled in the course.
...
@@ -531,7 +531,7 @@ def iterate_grades_for(course_or_id, students):
...
@@ -531,7 +531,7 @@ def iterate_grades_for(course_or_id, students):
# It's not pretty, but untangling that is currently beyond the
# It's not pretty, but untangling that is currently beyond the
# scope of this feature.
# scope of this feature.
request
.
session
=
{}
request
.
session
=
{}
gradeset
=
grade
(
student
,
request
,
course
)
gradeset
=
grade
(
student
,
request
,
course
,
keep_raw_scores
)
yield
student
,
gradeset
,
""
yield
student
,
gradeset
,
""
except
Exception
as
exc
:
# pylint: disable=broad-except
except
Exception
as
exc
:
# pylint: disable=broad-except
# Keep marching on even if this student couldn't be graded for
# Keep marching on even if this student couldn't be graded for
...
...
lms/djangoapps/courseware/tests/test_grades.py
View file @
9269ec3b
...
@@ -68,7 +68,7 @@ class TestGradeIteration(ModuleStoreTestCase):
...
@@ -68,7 +68,7 @@ class TestGradeIteration(ModuleStoreTestCase):
def
test_all_empty_grades
(
self
):
def
test_all_empty_grades
(
self
):
"""No students have grade entries"""
"""No students have grade entries"""
all_gradesets
,
all_errors
=
self
.
_gradesets_and_errors_for
(
self
.
course
.
id
,
self
.
students
)
all_gradesets
,
all_errors
=
self
.
_gradesets_and_errors_for
(
self
.
course
.
id
,
self
.
students
,
keep_raw_scores
=
True
)
self
.
assertEqual
(
len
(
all_errors
),
0
)
self
.
assertEqual
(
len
(
all_errors
),
0
)
for
gradeset
in
all_gradesets
.
values
():
for
gradeset
in
all_gradesets
.
values
():
self
.
assertIsNone
(
gradeset
[
'grade'
])
self
.
assertIsNone
(
gradeset
[
'grade'
])
...
@@ -107,7 +107,7 @@ class TestGradeIteration(ModuleStoreTestCase):
...
@@ -107,7 +107,7 @@ class TestGradeIteration(ModuleStoreTestCase):
self
.
assertTrue
(
all_gradesets
[
student5
])
self
.
assertTrue
(
all_gradesets
[
student5
])
################################# Helpers #################################
################################# Helpers #################################
def
_gradesets_and_errors_for
(
self
,
course_id
,
students
):
def
_gradesets_and_errors_for
(
self
,
course_id
,
students
,
keep_raw_scores
=
False
):
"""Simple helper method to iterate through student grades and give us
"""Simple helper method to iterate through student grades and give us
two dictionaries -- one that has all students and their respective
two dictionaries -- one that has all students and their respective
gradesets, and one that has only students that could not be graded and
gradesets, and one that has only students that could not be graded and
...
@@ -115,7 +115,7 @@ class TestGradeIteration(ModuleStoreTestCase):
...
@@ -115,7 +115,7 @@ class TestGradeIteration(ModuleStoreTestCase):
students_to_gradesets
=
{}
students_to_gradesets
=
{}
students_to_errors
=
{}
students_to_errors
=
{}
for
student
,
gradeset
,
err_msg
in
iterate_grades_for
(
course_id
,
students
):
for
student
,
gradeset
,
err_msg
in
iterate_grades_for
(
course_id
,
students
,
keep_raw_scores
):
students_to_gradesets
[
student
]
=
gradeset
students_to_gradesets
[
student
]
=
gradeset
if
err_msg
:
if
err_msg
:
students_to_errors
[
student
]
=
err_msg
students_to_errors
[
student
]
=
err_msg
...
...
lms/djangoapps/django_comment_client/tests/utils.py
View file @
9269ec3b
...
@@ -76,6 +76,15 @@ class ContentGroupTestCase(ModuleStoreTestCase):
...
@@ -76,6 +76,15 @@ class ContentGroupTestCase(ModuleStoreTestCase):
scheme_id
=
'cohort'
scheme_id
=
'cohort'
)
)
],
],
grading_policy
=
{
"GRADER"
:
[{
"type"
:
"Homework"
,
"min_count"
:
1
,
"drop_count"
:
0
,
"short_label"
:
"HW"
,
"weight"
:
1.0
}]
},
cohort_config
=
{
'cohorted'
:
True
},
cohort_config
=
{
'cohorted'
:
True
},
discussion_topics
=
{}
discussion_topics
=
{}
)
)
...
...
lms/djangoapps/instructor/views/api.py
View file @
9269ec3b
...
@@ -1954,6 +1954,34 @@ def calculate_grades_csv(request, course_id):
...
@@ -1954,6 +1954,34 @@ def calculate_grades_csv(request, course_id):
@ensure_csrf_cookie
@ensure_csrf_cookie
@cache_control
(
no_cache
=
True
,
no_store
=
True
,
must_revalidate
=
True
)
@cache_control
(
no_cache
=
True
,
no_store
=
True
,
must_revalidate
=
True
)
@require_level
(
'staff'
)
@require_level
(
'staff'
)
def
problem_grade_report
(
request
,
course_id
):
"""
Request a CSV showing students' weighted grades for all problems in the
course.
AlreadyRunningError is raised if the course's grades are already being
updated.
"""
course_key
=
SlashSeparatedCourseKey
.
from_deprecated_string
(
course_id
)
try
:
instructor_task
.
api
.
submit_problem_grade_report
(
request
,
course_key
)
# TODO: verify copy with documentation team
success_status
=
_
(
"Your weighted problem report is being generated! "
"You can view the status of the generation task in the 'Pending Instructor Tasks' section."
)
return
JsonResponse
({
"status"
:
success_status
})
except
AlreadyRunningError
:
# TODO: verify copy with documentation team
already_running_status
=
_
(
"A weighted problem generation task is already in progress. "
"Check the 'Pending Instructor Tasks' table for the status of the task. "
"When completed, the report will be available for download in the table below."
)
return
JsonResponse
({
"status"
:
already_running_status
})
@ensure_csrf_cookie
@cache_control
(
no_cache
=
True
,
no_store
=
True
,
must_revalidate
=
True
)
@require_level
(
'staff'
)
@require_query_params
(
'rolename'
)
@require_query_params
(
'rolename'
)
def
list_forum_members
(
request
,
course_id
):
def
list_forum_members
(
request
,
course_id
):
"""
"""
...
...
lms/djangoapps/instructor/views/api_urls.py
View file @
9269ec3b
...
@@ -87,6 +87,8 @@ urlpatterns = patterns(
...
@@ -87,6 +87,8 @@ urlpatterns = patterns(
'instructor.views.api.list_report_downloads'
,
name
=
"list_report_downloads"
),
'instructor.views.api.list_report_downloads'
,
name
=
"list_report_downloads"
),
url
(
r'calculate_grades_csv$'
,
url
(
r'calculate_grades_csv$'
,
'instructor.views.api.calculate_grades_csv'
,
name
=
"calculate_grades_csv"
),
'instructor.views.api.calculate_grades_csv'
,
name
=
"calculate_grades_csv"
),
url
(
r'problem_grade_report$'
,
'instructor.views.api.problem_grade_report'
,
name
=
"problem_grade_report"
),
# Registration Codes..
# Registration Codes..
url
(
r'get_registration_codes$'
,
url
(
r'get_registration_codes$'
,
...
...
lms/djangoapps/instructor/views/instructor_dashboard.py
View file @
9269ec3b
...
@@ -420,6 +420,7 @@ def _section_data_download(course, access):
...
@@ -420,6 +420,7 @@ def _section_data_download(course, access):
'list_instructor_tasks_url'
:
reverse
(
'list_instructor_tasks'
,
kwargs
=
{
'course_id'
:
unicode
(
course_key
)}),
'list_instructor_tasks_url'
:
reverse
(
'list_instructor_tasks'
,
kwargs
=
{
'course_id'
:
unicode
(
course_key
)}),
'list_report_downloads_url'
:
reverse
(
'list_report_downloads'
,
kwargs
=
{
'course_id'
:
unicode
(
course_key
)}),
'list_report_downloads_url'
:
reverse
(
'list_report_downloads'
,
kwargs
=
{
'course_id'
:
unicode
(
course_key
)}),
'calculate_grades_csv_url'
:
reverse
(
'calculate_grades_csv'
,
kwargs
=
{
'course_id'
:
unicode
(
course_key
)}),
'calculate_grades_csv_url'
:
reverse
(
'calculate_grades_csv'
,
kwargs
=
{
'course_id'
:
unicode
(
course_key
)}),
'problem_grade_report_url'
:
reverse
(
'problem_grade_report'
,
kwargs
=
{
'course_id'
:
unicode
(
course_key
)}),
}
}
return
section_data
return
section_data
...
...
lms/djangoapps/instructor_task/api.py
View file @
9269ec3b
...
@@ -19,6 +19,7 @@ from instructor_task.tasks import (
...
@@ -19,6 +19,7 @@ from instructor_task.tasks import (
delete_problem_state
,
delete_problem_state
,
send_bulk_course_email
,
send_bulk_course_email
,
calculate_grades_csv
,
calculate_grades_csv
,
calculate_problem_grade_report
,
calculate_students_features_csv
,
calculate_students_features_csv
,
cohort_students
,
cohort_students
,
)
)
...
@@ -334,6 +335,18 @@ def submit_calculate_grades_csv(request, course_key):
...
@@ -334,6 +335,18 @@ def submit_calculate_grades_csv(request, course_key):
return
submit_task
(
request
,
task_type
,
task_class
,
course_key
,
task_input
,
task_key
)
return
submit_task
(
request
,
task_type
,
task_class
,
course_key
,
task_input
,
task_key
)
def
submit_problem_grade_report
(
request
,
course_key
):
"""
Submits a task to generate a CSV grade report containing weighted problem
values.
"""
task_type
=
'grade_problems'
task_class
=
calculate_problem_grade_report
task_input
=
{}
task_key
=
""
return
submit_task
(
request
,
task_type
,
task_class
,
course_key
,
task_input
,
task_key
)
def
submit_calculate_students_features_csv
(
request
,
course_key
,
features
):
def
submit_calculate_students_features_csv
(
request
,
course_key
,
features
):
"""
"""
Submits a task to generate a CSV containing student profile info.
Submits a task to generate a CSV containing student profile info.
...
...
lms/djangoapps/instructor_task/tasks.py
View file @
9269ec3b
...
@@ -35,6 +35,7 @@ from instructor_task.tasks_helper import (
...
@@ -35,6 +35,7 @@ from instructor_task.tasks_helper import (
reset_attempts_module_state
,
reset_attempts_module_state
,
delete_problem_module_state
,
delete_problem_module_state
,
upload_grades_csv
,
upload_grades_csv
,
upload_problem_grade_report
,
upload_students_csv
,
upload_students_csv
,
cohort_students_and_upload
cohort_students_and_upload
)
)
...
@@ -155,6 +156,25 @@ def calculate_grades_csv(entry_id, xmodule_instance_args):
...
@@ -155,6 +156,25 @@ def calculate_grades_csv(entry_id, xmodule_instance_args):
return
run_main_task
(
entry_id
,
task_fn
,
action_name
)
return
run_main_task
(
entry_id
,
task_fn
,
action_name
)
# TODO: GRADES_DOWNLOAD_ROUTING_KEY is the high mem queue. Do we know we need it?
@task
(
base
=
BaseInstructorTask
,
routing_key
=
settings
.
GRADES_DOWNLOAD_ROUTING_KEY
)
# pylint: disable=not-callable
def
calculate_problem_grade_report
(
entry_id
,
xmodule_instance_args
):
"""
Generate a CSV for a course containing all students' weighted problem
grades and push the results to an S3 bucket for download.
"""
# Translators: This is a past-tense verb that is inserted into task progress messages as {action}.
# TODO: can this be the same as the `calculate_grades_csv` action_name?
action_name
=
ugettext_noop
(
'graded'
)
TASK_LOG
.
info
(
u"Task:
%
s, InstructorTask ID:
%
s, Task type:
%
s, Preparing for task execution"
,
xmodule_instance_args
.
get
(
'task_id'
),
entry_id
,
action_name
)
task_fn
=
partial
(
upload_problem_grade_report
,
xmodule_instance_args
)
return
run_main_task
(
entry_id
,
task_fn
,
action_name
)
@task
(
base
=
BaseInstructorTask
,
routing_key
=
settings
.
GRADES_DOWNLOAD_ROUTING_KEY
)
# pylint: disable=not-callable
@task
(
base
=
BaseInstructorTask
,
routing_key
=
settings
.
GRADES_DOWNLOAD_ROUTING_KEY
)
# pylint: disable=not-callable
def
calculate_students_features_csv
(
entry_id
,
xmodule_instance_args
):
def
calculate_students_features_csv
(
entry_id
,
xmodule_instance_args
):
"""
"""
...
...
lms/djangoapps/instructor_task/tasks_helper.py
View file @
9269ec3b
...
@@ -4,7 +4,10 @@ running state of a course.
...
@@ -4,7 +4,10 @@ running state of a course.
"""
"""
import
json
import
json
from
collections
import
OrderedDict
from
datetime
import
datetime
from
datetime
import
datetime
from
eventtracking
import
tracker
from
itertools
import
chain
from
time
import
time
from
time
import
time
import
unicodecsv
import
unicodecsv
import
logging
import
logging
...
@@ -34,6 +37,7 @@ from instructor_task.models import ReportStore, InstructorTask, PROGRESS
...
@@ -34,6 +37,7 @@ from instructor_task.models import ReportStore, InstructorTask, PROGRESS
from
lms.djangoapps.lms_xblock.runtime
import
LmsPartitionService
from
lms.djangoapps.lms_xblock.runtime
import
LmsPartitionService
from
openedx.core.djangoapps.course_groups.cohorts
import
get_cohort
from
openedx.core.djangoapps.course_groups.cohorts
import
get_cohort
from
openedx.core.djangoapps.course_groups.models
import
CourseUserGroup
from
openedx.core.djangoapps.course_groups.models
import
CourseUserGroup
from
openedx.core.djangoapps.content.course_structures.models
import
CourseStructure
from
opaque_keys.edx.keys
import
UsageKey
from
opaque_keys.edx.keys
import
UsageKey
from
openedx.core.djangoapps.course_groups.cohorts
import
add_user_to_cohort
,
is_course_cohorted
from
openedx.core.djangoapps.course_groups.cohorts
import
add_user_to_cohort
,
is_course_cohorted
from
student.models
import
CourseEnrollment
from
student.models
import
CourseEnrollment
...
@@ -705,6 +709,105 @@ def upload_grades_csv(_xmodule_instance_args, _entry_id, course_id, _task_input,
...
@@ -705,6 +709,105 @@ def upload_grades_csv(_xmodule_instance_args, _entry_id, course_id, _task_input,
return
task_progress
.
update_task_state
(
extra_meta
=
current_step
)
return
task_progress
.
update_task_state
(
extra_meta
=
current_step
)
def
_order_problems
(
blocks
):
"""
Sort the problems by the assignment type and assignment that it belongs to.
"""
problems
=
OrderedDict
()
assignments
=
dict
()
# First, sort out all the blocks into their correct assignments and all the
# assignments into their correct types.
for
block
in
blocks
:
# Put the assignments in order into the assignments list.
if
blocks
[
block
][
'block_type'
]
==
'sequential'
:
block_format
=
blocks
[
block
][
'format'
]
if
block_format
not
in
assignments
:
assignments
[
block_format
]
=
OrderedDict
()
assignments
[
block_format
][
block
]
=
list
()
# Put the problems into the correct order within their assignment.
if
blocks
[
block
][
'block_type'
]
==
'problem'
and
blocks
[
block
][
'graded'
]
is
True
:
current
=
blocks
[
block
][
'parent'
]
# crawl up the tree for the sequential block
while
blocks
[
current
][
'block_type'
]
!=
'sequential'
:
current
=
blocks
[
current
][
'parent'
]
current_format
=
blocks
[
current
][
'format'
]
assignments
[
current_format
][
current
]
.
append
(
block
)
# Now that we have a sorting and an order for the assignments and problems,
# iterate through them in order to generate the header row.
for
assignment_type
in
assignments
:
for
assignment_index
,
assignment
in
enumerate
(
assignments
[
assignment_type
]
.
keys
(),
start
=
1
):
for
problem
in
assignments
[
assignment_type
][
assignment
]:
header_name
=
"{assignment_type} {assignment_index}: {assignment_name} - {block}"
.
format
(
block
=
blocks
[
problem
][
'display_name'
],
assignment_type
=
assignment_type
,
assignment_index
=
assignment_index
,
assignment_name
=
blocks
[
assignment
][
'display_name'
]
)
problems
[
problem
]
=
[
header_name
+
" (Earned)"
,
header_name
+
" (Possible)"
]
return
problems
def
upload_problem_grade_report
(
_xmodule_instance_args
,
_entry_id
,
course_id
,
_task_input
,
action_name
):
"""
Generate a CSV containing all students' problem grades within a given
`course_id`.
"""
start_time
=
time
()
start_date
=
datetime
.
now
(
UTC
)
status_interval
=
100
enrolled_students
=
CourseEnrollment
.
users_enrolled_in
(
course_id
)
task_progress
=
TaskProgress
(
action_name
,
enrolled_students
.
count
(),
start_time
)
# This struct encapsulates both the display names of each static item in
# the header row as values as well as the django User field names of those
# items as the keys. It is structured in this way to keep the values
# related.
header_row
=
OrderedDict
([(
'id'
,
'Student ID'
),
(
'email'
,
'Email'
),
(
'username'
,
'Username'
)])
try
:
course_structure
=
CourseStructure
.
objects
.
get
(
course_id
=
course_id
)
blocks
=
course_structure
.
ordered_blocks
problems
=
_order_problems
(
blocks
)
except
CourseStructure
.
DoesNotExist
:
return
task_progress
.
update_task_state
(
extra_meta
=
{
'step'
:
'Generating course structure. Please refresh and try again.'
})
# Just generate the static fields for now.
rows
=
[
list
(
header_row
.
values
())
+
[
'Final Grade'
]
+
list
(
chain
.
from_iterable
(
problems
.
values
()))]
current_step
=
{
'step'
:
'Calculating Grades'
}
for
student
,
gradeset
,
err_msg
in
iterate_grades_for
(
course_id
,
enrolled_students
,
keep_raw_scores
=
True
):
student_fields
=
[
getattr
(
student
,
field_name
)
for
field_name
in
header_row
]
final_grade
=
gradeset
[
'percent'
]
# Only consider graded problems
problem_scores
=
{
unicode
(
score
.
module_id
):
score
for
score
in
gradeset
[
'raw_scores'
]
if
score
.
graded
}
earned_possible_values
=
list
()
for
problem_id
in
problems
:
try
:
problem_score
=
problem_scores
[
problem_id
]
earned_possible_values
.
append
([
problem_score
.
earned
,
problem_score
.
possible
])
except
KeyError
:
# The student has not been graded on this problem. For example,
# iterate_grades_for skips problems that students have never
# seen in order to speed up report generation. It could also be
# the case that the student does not have access to it (e.g. A/B
# test or cohorted courseware).
earned_possible_values
.
append
([
'N/A'
,
'N/A'
])
rows
.
append
(
student_fields
+
[
final_grade
]
+
list
(
chain
.
from_iterable
(
earned_possible_values
)))
task_progress
.
attempted
+=
1
task_progress
.
succeeded
+=
1
if
task_progress
.
attempted
%
status_interval
==
0
:
task_progress
.
update_task_state
(
extra_meta
=
current_step
)
# Perform the upload
upload_csv_to_report_store
(
rows
,
'problem_grade_report'
,
course_id
,
start_date
)
return
task_progress
.
update_task_state
(
extra_meta
=
{
'step'
:
'Uploading CSV'
})
def
upload_students_csv
(
_xmodule_instance_args
,
_entry_id
,
course_id
,
task_input
,
action_name
):
def
upload_students_csv
(
_xmodule_instance_args
,
_entry_id
,
course_id
,
task_input
,
action_name
):
"""
"""
For a given `course_id`, generate a CSV file containing profile
For a given `course_id`, generate a CSV file containing profile
...
...
lms/djangoapps/instructor_task/tests/test_base.py
View file @
9269ec3b
...
@@ -127,7 +127,9 @@ class InstructorTaskCourseTestCase(LoginEnrollmentTestCase, ModuleStoreTestCase)
...
@@ -127,7 +127,9 @@ class InstructorTaskCourseTestCase(LoginEnrollmentTestCase, ModuleStoreTestCase)
if
course_factory_kwargs
is
not
None
:
if
course_factory_kwargs
is
not
None
:
course_args
.
update
(
course_factory_kwargs
)
course_args
.
update
(
course_factory_kwargs
)
self
.
course
=
CourseFactory
.
create
(
**
course_args
)
self
.
course
=
CourseFactory
.
create
(
**
course_args
)
self
.
add_course_content
()
def
add_course_content
(
self
):
# Add a chapter to the course
# Add a chapter to the course
chapter
=
ItemFactory
.
create
(
parent_location
=
self
.
course
.
location
,
chapter
=
ItemFactory
.
create
(
parent_location
=
self
.
course
.
location
,
display_name
=
TEST_SECTION_NAME
)
display_name
=
TEST_SECTION_NAME
)
...
@@ -141,12 +143,13 @@ class InstructorTaskCourseTestCase(LoginEnrollmentTestCase, ModuleStoreTestCase)
...
@@ -141,12 +143,13 @@ class InstructorTaskCourseTestCase(LoginEnrollmentTestCase, ModuleStoreTestCase)
@staticmethod
@staticmethod
def
get_user_email
(
username
):
def
get_user_email
(
username
):
"""Generate email address based on username"""
"""Generate email address based on username"""
return
'{0}@test.com'
.
format
(
username
)
return
u
'{0}@test.com'
.
format
(
username
)
def
login_username
(
self
,
username
):
def
login_username
(
self
,
username
):
"""Login the user, given the `username`."""
"""Login the user, given the `username`."""
if
self
.
current_user
!=
username
:
if
self
.
current_user
!=
username
:
self
.
login
(
InstructorTaskCourseTestCase
.
get_user_email
(
username
),
"test"
)
user_email
=
User
.
objects
.
get
(
username
=
username
)
.
email
self
.
login
(
user_email
,
"test"
)
self
.
current_user
=
username
self
.
current_user
=
username
def
_create_user
(
self
,
username
,
email
=
None
,
is_staff
=
False
,
mode
=
'honor'
):
def
_create_user
(
self
,
username
,
email
=
None
,
is_staff
=
False
,
mode
=
'honor'
):
...
@@ -190,16 +193,18 @@ class InstructorTaskModuleTestCase(InstructorTaskCourseTestCase):
...
@@ -190,16 +193,18 @@ class InstructorTaskModuleTestCase(InstructorTaskCourseTestCase):
the setup of a course and problem in order to access StudentModule state.
the setup of a course and problem in order to access StudentModule state.
"""
"""
@staticmethod
@staticmethod
def
problem_location
(
problem_url_name
):
def
problem_location
(
problem_url_name
,
course_key
=
None
):
"""
"""
Create an internal location for a test problem.
Create an internal location for a test problem.
"""
"""
if
"i4x:"
in
problem_url_name
:
if
"i4x:"
in
problem_url_name
:
return
Location
.
from_deprecated_string
(
problem_url_name
)
return
Location
.
from_deprecated_string
(
problem_url_name
)
elif
course_key
:
return
course_key
.
make_usage_key
(
'problem'
,
problem_url_name
)
else
:
else
:
return
TEST_COURSE_KEY
.
make_usage_key
(
'problem'
,
problem_url_name
)
return
TEST_COURSE_KEY
.
make_usage_key
(
'problem'
,
problem_url_name
)
def
define_option_problem
(
self
,
problem_url_name
,
parent
=
None
):
def
define_option_problem
(
self
,
problem_url_name
,
parent
=
None
,
**
kwargs
):
"""Create the problem definition so the answer is Option 1"""
"""Create the problem definition so the answer is Option 1"""
if
parent
is
None
:
if
parent
is
None
:
parent
=
self
.
problem_section
parent
=
self
.
problem_section
...
@@ -213,7 +218,8 @@ class InstructorTaskModuleTestCase(InstructorTaskCourseTestCase):
...
@@ -213,7 +218,8 @@ class InstructorTaskModuleTestCase(InstructorTaskCourseTestCase):
parent
=
parent
,
parent
=
parent
,
category
=
"problem"
,
category
=
"problem"
,
display_name
=
str
(
problem_url_name
),
display_name
=
str
(
problem_url_name
),
data
=
problem_xml
)
data
=
problem_xml
,
**
kwargs
)
def
redefine_option_problem
(
self
,
problem_url_name
):
def
redefine_option_problem
(
self
,
problem_url_name
):
"""Change the problem definition so the answer is Option 2"""
"""Change the problem definition so the answer is Option 2"""
...
@@ -249,8 +255,9 @@ class InstructorTaskModuleTestCase(InstructorTaskCourseTestCase):
...
@@ -249,8 +255,9 @@ class InstructorTaskModuleTestCase(InstructorTaskCourseTestCase):
# Note that this is a capa-specific convention. The form is a version of the problem's
# Note that this is a capa-specific convention. The form is a version of the problem's
# URL, modified so that it can be easily stored in html, prepended with "input-" and
# URL, modified so that it can be easily stored in html, prepended with "input-" and
# appended with a sequence identifier for the particular response the input goes to.
# appended with a sequence identifier for the particular response the input goes to.
return
'input_i4x-{0}-{1}-problem-{2}_{3}'
.
format
(
TEST_COURSE_ORG
.
lower
(),
course_key
=
self
.
course
.
id
TEST_COURSE_NUMBER
.
replace
(
'.'
,
'_'
),
return
'input_i4x-{0}-{1}-problem-{2}_{3}'
.
format
(
course_key
.
org
,
course_key
.
course
.
replace
(
'.'
,
'_'
),
problem_url_name
,
response_id
)
problem_url_name
,
response_id
)
# make sure that the requested user is logged in, so that the ajax call works
# make sure that the requested user is logged in, so that the ajax call works
...
@@ -260,7 +267,7 @@ class InstructorTaskModuleTestCase(InstructorTaskCourseTestCase):
...
@@ -260,7 +267,7 @@ class InstructorTaskModuleTestCase(InstructorTaskCourseTestCase):
modx_url
=
reverse
(
'xblock_handler'
,
kwargs
=
{
modx_url
=
reverse
(
'xblock_handler'
,
kwargs
=
{
'course_id'
:
self
.
course
.
id
.
to_deprecated_string
(),
'course_id'
:
self
.
course
.
id
.
to_deprecated_string
(),
'usage_id'
:
quote_slashes
(
'usage_id'
:
quote_slashes
(
InstructorTaskModuleTestCase
.
problem_location
(
problem_url_name
)
.
to_deprecated_string
()
InstructorTaskModuleTestCase
.
problem_location
(
problem_url_name
,
self
.
course
.
id
)
.
to_deprecated_string
()
),
),
'handler'
:
'xmodule_handler'
,
'handler'
:
'xmodule_handler'
,
'suffix'
:
'problem_check'
,
'suffix'
:
'problem_check'
,
...
...
lms/djangoapps/instructor_task/tests/test_integration.py
View file @
9269ec3b
...
@@ -28,7 +28,7 @@ from instructor_task.api import (submit_rescore_problem_for_all_students,
...
@@ -28,7 +28,7 @@ from instructor_task.api import (submit_rescore_problem_for_all_students,
submit_reset_problem_attempts_for_all_students
,
submit_reset_problem_attempts_for_all_students
,
submit_delete_problem_state_for_all_students
)
submit_delete_problem_state_for_all_students
)
from
instructor_task.models
import
InstructorTask
from
instructor_task.models
import
InstructorTask
from
instructor_task.tasks_helper
import
upload_grades_csv
from
instructor_task.tasks_helper
import
upload_grades_csv
,
upload_problem_grade_report
from
instructor_task.tests.test_base
import
(
InstructorTaskModuleTestCase
,
TestReportMixin
,
TEST_COURSE_ORG
,
from
instructor_task.tests.test_base
import
(
InstructorTaskModuleTestCase
,
TestReportMixin
,
TEST_COURSE_ORG
,
TEST_COURSE_NUMBER
,
OPTION_1
,
OPTION_2
)
TEST_COURSE_NUMBER
,
OPTION_1
,
OPTION_2
)
from
capa.responsetypes
import
StudentInputError
from
capa.responsetypes
import
StudentInputError
...
...
lms/djangoapps/instructor_task/tests/test_tasks_helper.py
View file @
9269ec3b
...
@@ -19,6 +19,7 @@ from instructor_task.tasks_helper import cohort_students_and_upload, upload_grad
...
@@ -19,6 +19,7 @@ from instructor_task.tasks_helper import cohort_students_and_upload, upload_grad
from
instructor_task.tests.test_base
import
InstructorTaskCourseTestCase
,
TestReportMixin
,
InstructorTaskModuleTestCase
from
instructor_task.tests.test_base
import
InstructorTaskCourseTestCase
,
TestReportMixin
,
InstructorTaskModuleTestCase
from
openedx.core.djangoapps.course_groups.models
import
CourseUserGroupPartitionGroup
from
openedx.core.djangoapps.course_groups.models
import
CourseUserGroupPartitionGroup
from
openedx.core.djangoapps.course_groups.tests.helpers
import
CohortFactory
from
openedx.core.djangoapps.course_groups.tests.helpers
import
CohortFactory
from
openedx.core.djangoapps.user_api.tests.factories
import
UserCourseTagFactory
import
openedx.core.djangoapps.user_api.course_tag.api
as
course_tag_api
import
openedx.core.djangoapps.user_api.course_tag.api
as
course_tag_api
from
openedx.core.djangoapps.user_api.partition_schemes
import
RandomUserPartitionScheme
from
openedx.core.djangoapps.user_api.partition_schemes
import
RandomUserPartitionScheme
from
student.tests.factories
import
UserFactory
from
student.tests.factories
import
UserFactory
...
@@ -26,6 +27,13 @@ from student.models import CourseEnrollment
...
@@ -26,6 +27,13 @@ from student.models import CourseEnrollment
from
verify_student.tests.factories
import
SoftwareSecurePhotoVerificationFactory
from
verify_student.tests.factories
import
SoftwareSecurePhotoVerificationFactory
from
xmodule.modulestore.tests.factories
import
CourseFactory
,
ItemFactory
from
xmodule.modulestore.tests.factories
import
CourseFactory
,
ItemFactory
from
xmodule.partitions.partitions
import
Group
,
UserPartition
from
xmodule.partitions.partitions
import
Group
,
UserPartition
from
instructor_task.models
import
ReportStore
from
instructor_task.tasks_helper
import
(
cohort_students_and_upload
,
upload_grades_csv
,
upload_problem_grade_report
,
upload_students_csv
)
from
instructor_task.tests.test_base
import
InstructorTaskCourseTestCase
,
TestReportMixin
from
instructor_task.tests.test_integration
import
TestGradeReportConditionalContent
from
django_comment_client.tests.utils
import
ContentGroupTestCase
@ddt.ddt
@ddt.ddt
...
@@ -261,6 +269,164 @@ class TestInstructorGradeReport(TestReportMixin, InstructorTaskCourseTestCase):
...
@@ -261,6 +269,164 @@ class TestInstructorGradeReport(TestReportMixin, InstructorTaskCourseTestCase):
self
.
assertDictContainsSubset
({
'attempted'
:
1
,
'succeeded'
:
1
,
'failed'
:
0
},
result
)
self
.
assertDictContainsSubset
({
'attempted'
:
1
,
'succeeded'
:
1
,
'failed'
:
0
},
result
)
class
TestProblemGradeReport
(
TestReportMixin
,
InstructorTaskModuleTestCase
):
"""
Test that the weighted problem CSV generation works.
"""
def
setUp
(
self
):
super
(
TestProblemGradeReport
,
self
)
.
setUp
()
self
.
maxDiff
=
None
self
.
initialize_course
()
# Add unicode data to CSV even though unicode usernames aren't
# technically possible in openedx.
self
.
student_1
=
self
.
create_student
(
u'üser_1'
)
self
.
student_2
=
self
.
create_student
(
u'üser_2'
)
self
.
csv_header_row
=
[
u'Student ID'
,
u'Email'
,
u'Username'
,
u'Final Grade'
]
@patch
(
'instructor_task.tasks_helper._get_current_task'
)
def
test_no_problems
(
self
,
_get_current_task
):
"""
Verify that we see no grade information for a course with no graded
problems.
"""
result
=
upload_problem_grade_report
(
None
,
None
,
self
.
course
.
id
,
None
,
'graded'
)
self
.
assertDictContainsSubset
({
'action_name'
:
'graded'
,
'attempted'
:
2
,
'succeeded'
:
2
,
'failed'
:
0
},
result
)
self
.
verify_rows_in_csv
([
dict
(
zip
(
self
.
csv_header_row
,
[
unicode
(
self
.
student_1
.
id
),
self
.
student_1
.
email
,
self
.
student_1
.
username
,
'0.0'
])),
dict
(
zip
(
self
.
csv_header_row
,
[
unicode
(
self
.
student_2
.
id
),
self
.
student_2
.
email
,
self
.
student_2
.
username
,
'0.0'
]))
])
@patch
(
'instructor_task.tasks_helper._get_current_task'
)
def
test_single_problem
(
self
,
_get_current_task
):
vertical
=
ItemFactory
.
create
(
parent_location
=
self
.
problem_section
.
location
,
category
=
'vertical'
,
metadata
=
{
'graded'
:
True
},
display_name
=
'Problem Vertical'
)
self
.
define_option_problem
(
'Problem1'
,
parent
=
vertical
)
# generate the course structure
self
.
submit_student_answer
(
self
.
student_1
.
username
,
'Problem1'
,
[
'Option 1'
])
result
=
upload_problem_grade_report
(
None
,
None
,
self
.
course
.
id
,
None
,
'graded'
)
self
.
assertDictContainsSubset
({
'action_name'
:
'graded'
,
'attempted'
:
2
,
'succeeded'
:
2
,
'failed'
:
0
},
result
)
problem_name
=
'Homework 1: Problem - Problem1'
header_row
=
self
.
csv_header_row
+
[
problem_name
+
' (Earned)'
,
problem_name
+
' (Possible)'
]
self
.
verify_rows_in_csv
([
dict
(
zip
(
header_row
,
[
unicode
(
self
.
student_1
.
id
),
self
.
student_1
.
email
,
self
.
student_1
.
username
,
'0.01'
,
'1.0'
,
'2.0'
]
)),
dict
(
zip
(
header_row
,
[
unicode
(
self
.
student_2
.
id
),
self
.
student_2
.
email
,
self
.
student_2
.
username
,
'0.0'
,
'N/A'
,
'N/A'
]
))
])
class
TestProblemReportSplitTestContent
(
TestGradeReportConditionalContent
):
OPTION_1
=
'Option 1'
OPTION_2
=
'Option 2'
def
setUp
(
self
):
super
(
TestProblemReportSplitTestContent
,
self
)
.
setUp
()
self
.
problem_a_url
=
'problem_a_url'
self
.
problem_b_url
=
'problem_b_url'
self
.
define_option_problem
(
self
.
problem_a_url
,
parent
=
self
.
vertical_a
)
self
.
define_option_problem
(
self
.
problem_b_url
,
parent
=
self
.
vertical_b
)
def
test_problem_grade_report
(
self
):
"""
Test problems that exist in a problem grade report.
"""
# student A will get 100%, student B will get 50% because
# OPTION_1 is the correct option, and OPTION_2 is the
# incorrect option
self
.
submit_student_answer
(
self
.
student_a
.
username
,
self
.
problem_a_url
,
[
self
.
OPTION_1
,
self
.
OPTION_1
])
self
.
submit_student_answer
(
self
.
student_a
.
username
,
self
.
problem_b_url
,
[
self
.
OPTION_1
,
self
.
OPTION_1
])
self
.
submit_student_answer
(
self
.
student_b
.
username
,
self
.
problem_a_url
,
[
self
.
OPTION_1
,
self
.
OPTION_2
])
self
.
submit_student_answer
(
self
.
student_b
.
username
,
self
.
problem_b_url
,
[
self
.
OPTION_1
,
self
.
OPTION_2
])
with
patch
(
'instructor_task.tasks_helper._get_current_task'
):
result
=
upload_problem_grade_report
(
None
,
None
,
self
.
course
.
id
,
None
,
'graded'
)
self
.
verify_csv_task_success
(
result
)
problem_names
=
[
'Homework 1: Problem - problem_a_url'
,
'Homework 1: Problem - problem_b_url'
]
header_row
=
[
u'Student ID'
,
u'Email'
,
u'Username'
,
u'Final Grade'
]
for
problem
in
problem_names
:
header_row
+=
[
problem
+
' (Earned)'
,
problem
+
' (Possible)'
]
self
.
verify_rows_in_csv
([
dict
(
zip
(
header_row
,
[
unicode
(
self
.
student_a
.
id
),
self
.
student_a
.
email
,
self
.
student_a
.
username
,
u'1.0'
,
u'2.0'
,
u'2.0'
,
u'N/A'
,
u'N/A'
]
)),
dict
(
zip
(
header_row
,
[
unicode
(
self
.
student_b
.
id
),
self
.
student_b
.
email
,
self
.
student_b
.
username
,
u'0.5'
,
u'N/A'
,
u'N/A'
,
u'1.0'
,
u'2.0'
]
))
])
class
TestProblemReportCohortedContent
(
TestReportMixin
,
ContentGroupTestCase
,
InstructorTaskModuleTestCase
):
def
setUp
(
self
):
super
(
TestProblemReportCohortedContent
,
self
)
.
setUp
()
# contstruct cohorted problems to work on.
self
.
add_course_content
()
vertical
=
ItemFactory
.
create
(
parent_location
=
self
.
problem_section
.
location
,
category
=
'vertical'
,
metadata
=
{
'graded'
:
True
},
display_name
=
'Problem Vertical'
)
print
self
.
course
.
user_partitions
self
.
define_option_problem
(
"Problem0"
,
parent
=
vertical
,
group_access
=
{
self
.
course
.
user_partitions
[
0
]
.
id
:
[
self
.
course
.
user_partitions
[
0
]
.
groups
[
0
]
.
id
]}
)
self
.
define_option_problem
(
"Problem1"
,
parent
=
vertical
,
group_access
=
{
self
.
course
.
user_partitions
[
0
]
.
id
:
[
self
.
course
.
user_partitions
[
0
]
.
groups
[
1
]
.
id
]}
)
self
.
submit_student_answer
(
self
.
alpha_user
.
username
,
'Problem0'
,
[
'Option 1'
,
'Option 1'
])
self
.
submit_student_answer
(
self
.
alpha_user
.
username
,
'Problem1'
,
[
'Option 1'
,
'Option 1'
])
self
.
submit_student_answer
(
self
.
beta_user
.
username
,
'Problem0'
,
[
'Option 1'
,
'Option 2'
])
self
.
submit_student_answer
(
self
.
beta_user
.
username
,
'Problem1'
,
[
'Option 1'
,
'Option 2'
])
def
test_cohort_content
(
self
):
with
patch
(
'instructor_task.tasks_helper._get_current_task'
):
result
=
upload_problem_grade_report
(
None
,
None
,
self
.
course
.
id
,
None
,
'graded'
)
self
.
assertDictContainsSubset
({
'action_name'
:
'graded'
,
'attempted'
:
4
,
'succeeded'
:
4
,
'failed'
:
0
},
result
)
problem_names
=
[
'Homework 1: Problem - Problem0'
,
'Homework 1: Problem - Problem1'
]
header_row
=
[
u'Student ID'
,
u'Email'
,
u'Username'
,
u'Final Grade'
]
for
problem
in
problem_names
:
header_row
+=
[
problem
+
' (Earned)'
,
problem
+
' (Possible)'
]
self
.
verify_rows_in_csv
([
dict
(
zip
(
header_row
,
[
unicode
(
self
.
staff_user
.
id
),
self
.
staff_user
.
email
,
self
.
staff_user
.
username
,
u'0.0'
,
u'N/A'
,
u'N/A'
,
u'N/A'
,
u'N/A'
]
)),
dict
(
zip
(
header_row
,
[
unicode
(
self
.
alpha_user
.
id
),
self
.
alpha_user
.
email
,
self
.
alpha_user
.
username
,
u'1.0'
,
u'2.0'
,
u'2.0'
,
u'N/A'
,
u'N/A'
]
)),
dict
(
zip
(
header_row
,
[
unicode
(
self
.
beta_user
.
id
),
self
.
beta_user
.
email
,
self
.
beta_user
.
username
,
u'0.5'
,
u'N/A'
,
u'N/A'
,
u'1.0'
,
u'2.0'
]
)),
dict
(
zip
(
header_row
,
[
unicode
(
self
.
non_cohorted_user
.
id
),
self
.
non_cohorted_user
.
email
,
self
.
non_cohorted_user
.
username
,
u'0.0'
,
u'N/A'
,
u'N/A'
,
u'N/A'
,
u'N/A'
]
)),
])
@ddt.ddt
@ddt.ddt
class
TestStudentReport
(
TestReportMixin
,
InstructorTaskCourseTestCase
):
class
TestStudentReport
(
TestReportMixin
,
InstructorTaskCourseTestCase
):
"""
"""
...
...
lms/static/coffee/src/instructor_dashboard/data_download.coffee
View file @
9269ec3b
...
@@ -22,6 +22,7 @@ class DataDownload
...
@@ -22,6 +22,7 @@ class DataDownload
@
$list_anon_btn
=
@
$section
.
find
(
"input[name='list-anon-ids']'"
)
@
$list_anon_btn
=
@
$section
.
find
(
"input[name='list-anon-ids']'"
)
@
$grade_config_btn
=
@
$section
.
find
(
"input[name='dump-gradeconf']'"
)
@
$grade_config_btn
=
@
$section
.
find
(
"input[name='dump-gradeconf']'"
)
@
$calculate_grades_csv_btn
=
@
$section
.
find
(
"input[name='calculate-grades-csv']'"
)
@
$calculate_grades_csv_btn
=
@
$section
.
find
(
"input[name='calculate-grades-csv']'"
)
@
$problem_grade_report_csv_btn
=
@
$section
.
find
(
"input[name='problem-grade-report']'"
)
# response areas
# response areas
@
$download
=
@
$section
.
find
'.data-download-container'
@
$download
=
@
$section
.
find
'.data-download-container'
...
@@ -108,16 +109,22 @@ class DataDownload
...
@@ -108,16 +109,22 @@ class DataDownload
@
$download_display_text
.
html
data
[
'grading_config_summary'
]
@
$download_display_text
.
html
data
[
'grading_config_summary'
]
@
$calculate_grades_csv_btn
.
click
(
e
)
=>
@
$calculate_grades_csv_btn
.
click
(
e
)
=>
@
onClickGradeDownload
@
$calculate_grades_csv_btn
,
"Error generating grades. Please try again."
@
$problem_grade_report_csv_btn
.
click
(
e
)
=>
@
onClickGradeDownload
@
$problem_grade_report_csv_btn
,
"Error generating weighted problem report. Please try again."
onClickGradeDownload
:
(
button
,
errorMessage
)
->
# Clear any CSS styling from the request-response areas
# Clear any CSS styling from the request-response areas
#$(".msg-confirm").css({"display":"none"})
#$(".msg-confirm").css({"display":"none"})
#$(".msg-error").css({"display":"none"})
#$(".msg-error").css({"display":"none"})
@
clear_display
()
@
clear_display
()
url
=
@
$calculate_grades_csv_bt
n
.
data
'endpoint'
url
=
butto
n
.
data
'endpoint'
$
.
ajax
$
.
ajax
dataType
:
'json'
dataType
:
'json'
url
:
url
url
:
url
error
:
(
std_ajax_err
)
=>
error
:
(
std_ajax_err
)
=>
@
$reports_request_response_error
.
text
gettext
(
"Error generating grades. Please try again."
)
@
$reports_request_response_error
.
text
gettext
(
errorMessage
)
$
(
".msg-error"
).
css
({
"display"
:
"block"
})
$
(
".msg-error"
).
css
({
"display"
:
"block"
})
success
:
(
data
)
=>
success
:
(
data
)
=>
@
$reports_request_response
.
text
data
[
'status'
]
@
$reports_request_response
.
text
data
[
'status'
]
...
...
lms/templates/instructor/instructor_dashboard_2/data_download.html
View file @
9269ec3b
...
@@ -41,6 +41,8 @@
...
@@ -41,6 +41,8 @@
<p>
${_("Click to generate a CSV grade report for all currently enrolled students.")}
</p>
<p>
${_("Click to generate a CSV grade report for all currently enrolled students.")}
</p>
<p><input
type=
"button"
name=
"calculate-grades-csv"
value=
"${_("
Generate
Grade
Report
")}"
data-endpoint=
"${ section_data['calculate_grades_csv_url'] }"
/></p>
<p><input
type=
"button"
name=
"calculate-grades-csv"
value=
"${_("
Generate
Grade
Report
")}"
data-endpoint=
"${ section_data['calculate_grades_csv_url'] }"
/></p>
<p><input
type=
"button"
name=
"problem-grade-report"
value=
"${_("
Generate
Problem
Grade
Report
")}"
data-endpoint=
"${ section_data['problem_grade_report_url'] }"
/></p>
%endif
%endif
<div
class=
"request-response msg msg-confirm copy"
id=
"report-request-response"
></div>
<div
class=
"request-response msg msg-confirm copy"
id=
"report-request-response"
></div>
...
...
openedx/core/djangoapps/content/course_structures/models.py
View file @
9269ec3b
import
json
import
json
import
logging
import
logging
from
collections
import
OrderedDict
from
model_utils.models
import
TimeStampedModel
from
model_utils.models
import
TimeStampedModel
from
util.models
import
CompressedTextField
from
util.models
import
CompressedTextField
...
@@ -26,6 +27,29 @@ class CourseStructure(TimeStampedModel):
...
@@ -26,6 +27,29 @@ class CourseStructure(TimeStampedModel):
return
json
.
loads
(
self
.
structure_json
)
return
json
.
loads
(
self
.
structure_json
)
return
None
return
None
@property
def
ordered_blocks
(
self
):
if
self
.
structure
:
ordered_blocks
=
OrderedDict
()
self
.
_traverse_tree
(
self
.
structure
[
'root'
],
self
.
structure
[
'blocks'
],
ordered_blocks
)
return
ordered_blocks
def
_traverse_tree
(
self
,
block
,
unordered_structure
,
ordered_blocks
,
parent
=
None
):
"""
Traverses the tree and fills in the ordered_blocks OrderedDict with the blocks in
the order that they appear in the course.
"""
# find the dictionary entry for the current node
cur_block
=
unordered_structure
[
block
]
if
parent
:
cur_block
[
'parent'
]
=
parent
ordered_blocks
[
block
]
=
cur_block
for
child_node
in
cur_block
[
'children'
]:
self
.
_traverse_tree
(
child_node
,
unordered_structure
,
ordered_blocks
,
parent
=
block
)
# Signals must be imported in a file that is automatically loaded at app startup (e.g. models.py). We import them
# Signals must be imported in a file that is automatically loaded at app startup (e.g. models.py). We import them
# at the end of this file to avoid circular dependencies.
# at the end of this file to avoid circular dependencies.
import
signals
# pylint: disable=unused-import
import
signals
# pylint: disable=unused-import
openedx/core/djangoapps/content/course_structures/tests.py
View file @
9269ec3b
...
@@ -91,6 +91,40 @@ class CourseStructureTaskTests(ModuleStoreTestCase):
...
@@ -91,6 +91,40 @@ class CourseStructureTaskTests(ModuleStoreTestCase):
cs
=
CourseStructure
.
objects
.
create
(
course_id
=
self
.
course
.
id
,
structure_json
=
structure_json
)
cs
=
CourseStructure
.
objects
.
create
(
course_id
=
self
.
course
.
id
,
structure_json
=
structure_json
)
self
.
assertDictEqual
(
cs
.
structure
,
structure
)
self
.
assertDictEqual
(
cs
.
structure
,
structure
)
def
test_ordered_blocks
(
self
):
structure
=
{
'root'
:
'a/b/c'
,
'blocks'
:
{
'a/b/c'
:
{
'id'
:
'a/b/c'
,
'children'
:
[
'g/h/i'
]
},
'd/e/f'
:
{
'id'
:
'd/e/f'
,
'children'
:
[]
},
'g/h/i'
:
{
'id'
:
'h/j/k'
,
'children'
:
[
'j/k/l'
,
'd/e/f'
]
},
'j/k/l'
:
{
'id'
:
'j/k/l'
,
'children'
:
[]
}
}
}
in_order_blocks
=
[
'a/b/c'
,
'g/h/i'
,
'j/k/l'
,
'd/e/f'
]
structure_json
=
json
.
dumps
(
structure
)
cs
=
CourseStructure
.
objects
.
create
(
course_id
=
self
.
course
.
id
,
structure_json
=
structure_json
)
self
.
assertEqual
(
cs
.
ordered_blocks
.
keys
(),
in_order_blocks
)
def
test_block_with_missing_fields
(
self
):
def
test_block_with_missing_fields
(
self
):
"""
"""
The generator should continue to operate on blocks/XModule that do not have graded or format fields.
The generator should continue to operate on blocks/XModule that do not have graded or format fields.
...
...
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