Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
E
edx-analytics-data-api
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-analytics-data-api
Commits
7278f1c5
Commit
7278f1c5
authored
Dec 22, 2015
by
Dennis Jen
Committed by
Daniel Friedman
Apr 11, 2016
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
added secondary storting by attempt_ratio
parent
5d84d4c7
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
73 additions
and
29 deletions
+73
-29
analytics_data_api/v0/models.py
+46
-16
analytics_data_api/v0/tests/views/test_learners.py
+15
-8
analytics_data_api/v0/views/learners.py
+12
-5
No files found.
analytics_data_api/v0/models.py
View file @
7278f1c5
...
@@ -256,14 +256,26 @@ class RosterEntry(DocType):
...
@@ -256,14 +256,26 @@ class RosterEntry(DocType):
cohort
=
None
,
cohort
=
None
,
enrollment_mode
=
None
,
enrollment_mode
=
None
,
text_search
=
None
,
text_search
=
None
,
order_by
=
'username'
,
order_by
_fields
=
None
,
sort_order
=
'asc'
sort_order
_directions
=
None
):
):
"""
"""
Construct a search query for all users in `course_id` and return
Construct a search query for all users in `course_id` and return
the Search object. Raises `ValueError` if both `segments` and
the Search object.
`ignore_segments` are provided.
order_by_fields specifies the fields to order by with the first element of
the array as the primary sort. Similarly, sort_order_directions corresponds
to the direction of the sort. These arrays must be of the same size.
Raises `ValueError` if both `segments` and `ignore_segments` are provided.
"""
"""
if
order_by_fields
is
None
:
order_by_fields
=
[
'username'
]
if
sort_order_directions
is
None
:
sort_order_directions
=
[
'asc'
]
# Error handling
# Error handling
if
segments
and
ignore_segments
:
if
segments
and
ignore_segments
:
raise
ValueError
(
'Cannot combine `segments` and `ignore_segments` parameters.'
)
raise
ValueError
(
'Cannot combine `segments` and `ignore_segments` parameters.'
)
...
@@ -272,19 +284,29 @@ class RosterEntry(DocType):
...
@@ -272,19 +284,29 @@ class RosterEntry(DocType):
raise
ValueError
(
"segments/ignore_segments value '{segment}' must be one of: ({segments})"
.
format
(
raise
ValueError
(
"segments/ignore_segments value '{segment}' must be one of: ({segments})"
.
format
(
segment
=
segment
,
segments
=
', '
.
join
(
learner
.
SEGMENTS
)
segment
=
segment
,
segments
=
', '
.
join
(
learner
.
SEGMENTS
)
))
))
order_by_options
=
(
order_by_options
=
(
'username'
,
'email'
,
'discussions_contributed'
,
'problems_attempted'
,
'problems_completed'
,
'username'
,
'email'
,
'discussions_contributed'
,
'problems_attempted'
,
'problems_completed'
,
'attempt_ratio_order'
,
'videos_viewed'
'
problem_attempts_per_completed'
,
'
attempt_ratio_order'
,
'videos_viewed'
)
)
if
order_by
not
in
order_by_options
:
for
order_by
in
order_by_fields
:
raise
ValueError
(
"order_by value '{order_by}' must be one of: ({order_by_options})"
.
format
(
if
order_by
not
in
order_by_options
:
order_by
=
order_by
,
order_by_options
=
', '
.
join
(
order_by_options
)
raise
ValueError
(
"order_by value '{order_by}' must be one of: ({order_by_options})"
.
format
(
))
order_by
=
order_by
,
order_by_options
=
', '
.
join
(
order_by_options
)
))
sort_order_options
=
(
'asc'
,
'desc'
)
sort_order_options
=
(
'asc'
,
'desc'
)
if
sort_order
not
in
sort_order_options
:
for
sort_order
in
sort_order_directions
:
raise
ValueError
(
"sort_order value '{sort_order}' must be one of: ({sort_order_options})"
.
format
(
if
sort_order
not
in
sort_order_options
:
sort_order
=
sort_order
,
sort_order_options
=
', '
.
join
(
sort_order_options
)
raise
ValueError
(
"sort_order value '{sort_order}' must be one of: ({sort_order_options})"
.
format
(
))
sort_order
=
sort_order
,
sort_order_options
=
', '
.
join
(
sort_order_options
)
))
if
len
(
order_by_fields
)
!=
len
(
sort_order_directions
):
raise
ValueError
(
"The number of items in order_by_fields '{order_by_count}' must equal that of "
"sort_order_directions '{sort_order_count}'"
.
format
(
order_by_count
=
len
(
order_by_fields
),
sort_order_count
=
len
(
sort_order_directions
)
))
search
=
cls
.
search
()
search
=
cls
.
search
()
search
.
query
=
Q
(
'bool'
,
must
=
[
Q
(
'term'
,
course_id
=
course_id
)])
search
.
query
=
Q
(
'bool'
,
must
=
[
Q
(
'term'
,
course_id
=
course_id
)])
...
@@ -302,9 +324,17 @@ class RosterEntry(DocType):
...
@@ -302,9 +324,17 @@ class RosterEntry(DocType):
if
text_search
:
if
text_search
:
search
.
query
.
must
.
append
(
Q
(
'multi_match'
,
query
=
text_search
,
fields
=
[
'name'
,
'username'
,
'email'
]))
search
.
query
.
must
.
append
(
Q
(
'multi_match'
,
query
=
text_search
,
fields
=
[
'name'
,
'username'
,
'email'
]))
# Sorting
# construct the sort hierarchy
sort_term
=
order_by
if
sort_order
==
'asc'
else
'-{}'
.
format
(
order_by
)
sort_terms
=
[]
search
=
search
.
sort
(
sort_term
)
for
order_by
,
sort_order
in
zip
(
order_by_fields
,
sort_order_directions
):
term
=
{
order_by
:
{
'order'
:
sort_order
,
'missing'
:
'_last'
if
sort_order
==
'asc'
else
'_first'
,
# ordering of missing fields
}
}
sort_terms
.
append
(
term
)
search
=
search
.
sort
(
*
sort_terms
)
return
search
return
search
...
...
analytics_data_api/v0/tests/views/test_learners.py
View file @
7278f1c5
...
@@ -382,16 +382,23 @@ class LearnerListTests(LearnerAPITestMixin, VerifyCourseIdMixin, TestCaseWithAut
...
@@ -382,16 +382,23 @@ class LearnerListTests(LearnerAPITestMixin, VerifyCourseIdMixin, TestCaseWithAut
'videos_viewed'
,
'desc'
,
[{
'username'
:
'b'
},
{
'username'
:
'a'
}]
'videos_viewed'
,
'desc'
,
[{
'username'
:
'b'
},
{
'username'
:
'a'
}]
),
),
(
(
[{
'username'
:
'a'
,
'problem_attempts_per_completed'
:
None
,
'attempt_ratio_order'
:
-
1
},
[{
'username'
:
'a'
,
'problem_attempts_per_completed'
:
1.0
,
'attempt_ratio_order'
:
1
},
{
'username'
:
'b'
,
'problem_attempts_per_completed'
:
None
,
'attempt_ratio_order'
:
0
},
{
'username'
:
'b'
,
'problem_attempts_per_completed'
:
2.0
,
'attempt_ratio_order'
:
10
},
{
'username'
:
'c'
,
'problem_attempts_per_completed'
:
None
,
'attempt_ratio_order'
:
1
}],
{
'username'
:
'c'
,
'problem_attempts_per_completed'
:
2.0
,
'attempt_ratio_order'
:
2
},
'problem_attempts_per_completed'
,
'asc'
,
[{
'username'
:
'a'
},
{
'username'
:
'b'
},
{
'username'
:
'c'
}]
{
'username'
:
'd'
,
'attempt_ratio_order'
:
0
},
{
'username'
:
'e'
,
'attempt_ratio_order'
:
-
10
}],
'problem_attempts_per_completed'
,
'asc'
,
[
{
'username'
:
'a'
},
{
'username'
:
'b'
},
{
'username'
:
'c'
},
{
'username'
:
'd'
},
{
'username'
:
'e'
}
]
),
),
(
(
[{
'username'
:
'a'
,
'problem_attempts_per_completed'
:
None
,
'attempt_ratio_order'
:
-
1
},
[{
'username'
:
'a'
,
'problem_attempts_per_completed'
:
1.0
,
'attempt_ratio_order'
:
1
},
{
'username'
:
'b'
,
'problem_attempts_per_completed'
:
0.0
,
'attempt_ratio_order'
:
0
},
{
'username'
:
'b'
,
'problem_attempts_per_completed'
:
2.0
,
'attempt_ratio_order'
:
10
},
{
'username'
:
'c'
,
'problem_attempts_per_completed'
:
123.64
,
'attempt_ratio_order'
:
1
}],
{
'username'
:
'c'
,
'problem_attempts_per_completed'
:
2.0
,
'attempt_ratio_order'
:
2
},
'problem_attempts_per_completed'
,
'desc'
,
[{
'username'
:
'c'
},
{
'username'
:
'b'
},
{
'username'
:
'a'
}]
{
'username'
:
'd'
,
'attempt_ratio_order'
:
0
},
{
'username'
:
'e'
,
'attempt_ratio_order'
:
-
10
}],
'problem_attempts_per_completed'
,
'desc'
,
[
{
'username'
:
'e'
},
{
'username'
:
'd'
},
{
'username'
:
'c'
},
{
'username'
:
'b'
},
{
'username'
:
'a'
}]
),
),
)
)
@ddt.unpack
@ddt.unpack
...
...
analytics_data_api/v0/views/learners.py
View file @
7278f1c5
...
@@ -175,11 +175,18 @@ class LearnerListView(CourseViewMixin, generics.ListAPIView):
...
@@ -175,11 +175,18 @@ class LearnerListView(CourseViewMixin, generics.ListAPIView):
self
.
_validate_query_params
()
self
.
_validate_query_params
()
query_params
=
self
.
request
.
QUERY_PARAMS
query_params
=
self
.
request
.
QUERY_PARAMS
order_by
=
query_params
.
get
(
'order_by'
)
order_by_fields
=
[
order_by
]
if
order_by
else
None
sort_order
=
query_params
.
get
(
'sort_order'
)
sort_order_directions
=
[
sort_order
]
if
sort_order
else
None
# Ordering by problem_attempts_per_completed can be ambiguous because
# Ordering by problem_attempts_per_completed can be ambiguous because
# values could be infinite (e.g. divide by zero) if no problems were completed.
# values could be infinite (e.g. divide by zero) if no problems were completed.
# Instead, sorting by attempt_ratio_order will produce a sensible ordering
# Instead, secondary sorting by attempt_ratio_order will produce a sensible ordering
order_by
=
query_params
.
get
(
'order_by'
)
if
order_by
==
'problem_attempts_per_completed'
:
order_by
=
'attempt_ratio_order'
if
order_by
==
'problem_attempts_per_completed'
else
order_by
order_by_fields
.
append
(
'attempt_ratio_order'
)
sort_order_directions
.
append
(
'asc'
if
sort_order
==
'desc'
else
'desc'
)
params
=
{
params
=
{
'segments'
:
split_query_argument
(
query_params
.
get
(
'segments'
)),
'segments'
:
split_query_argument
(
query_params
.
get
(
'segments'
)),
...
@@ -187,8 +194,8 @@ class LearnerListView(CourseViewMixin, generics.ListAPIView):
...
@@ -187,8 +194,8 @@ class LearnerListView(CourseViewMixin, generics.ListAPIView):
'cohort'
:
query_params
.
get
(
'cohort'
),
'cohort'
:
query_params
.
get
(
'cohort'
),
'enrollment_mode'
:
query_params
.
get
(
'enrollment_mode'
),
'enrollment_mode'
:
query_params
.
get
(
'enrollment_mode'
),
'text_search'
:
query_params
.
get
(
'text_search'
),
'text_search'
:
query_params
.
get
(
'text_search'
),
'order_by
'
:
order_by
,
'order_by
_fields'
:
order_by_fields
,
'sort_order
'
:
query_params
.
get
(
'sort_order'
)
'sort_order
_directions'
:
sort_order_directions
,
}
}
# Remove None values from `params` so that we don't overwrite default
# Remove None values from `params` so that we don't overwrite default
# parameter values in `get_users_in_course`.
# parameter values in `get_users_in_course`.
...
...
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