Commit 413bce3d by Jeff LaJoie

TNL-6739: adds in masquerading as enrollment track user

parent f00059f2
...@@ -36,22 +36,22 @@ class StaffPreviewPage(PageObject): ...@@ -36,22 +36,22 @@ class StaffPreviewPage(PageObject):
@property @property
def staff_view_mode(self): def staff_view_mode(self):
""" """
Return the currently chosen view mode, e.g. "Staff", "Student" or a content group. Return the currently chosen view mode, e.g. "Staff", "Learner" or a content group.
""" """
return self.q(css=self.VIEW_MODE_OPTIONS_CSS).filter(lambda el: el.is_selected()).first.text[0] return self.q(css=self.VIEW_MODE_OPTIONS_CSS).filter(lambda el: el.is_selected()).first.text[0]
def set_staff_view_mode(self, view_mode): def set_staff_view_mode(self, view_mode):
""" """
Set the current view mode, e.g. "Staff", "Student" or a content group. Set the current view mode, e.g. "Staff", "Learner" or a content group.
""" """
self.q(css=self.VIEW_MODE_OPTIONS_CSS).filter(lambda el: el.text.strip() == view_mode).first.click() self.q(css=self.VIEW_MODE_OPTIONS_CSS).filter(lambda el: el.text.strip() == view_mode).first.click()
self.wait_for_ajax() self.wait_for_ajax()
def set_staff_view_mode_specific_student(self, username_or_email): def set_staff_view_mode_specific_student(self, username_or_email):
""" """
Set the current preview mode to "Specific Student" with the given username or email Set the current preview mode to "Specific learner" with the given username or email
""" """
required_mode = "Specific student" required_mode = "Specific learner"
if self.staff_view_mode != required_mode: if self.staff_view_mode != required_mode:
self.q(css=self.VIEW_MODE_OPTIONS_CSS).filter(lambda el: el.text == required_mode).first.click() self.q(css=self.VIEW_MODE_OPTIONS_CSS).filter(lambda el: el.text == required_mode).first.click()
# Use a script here because .clear() + .send_keys() triggers unwanted behavior if a username is already set # Use a script here because .clear() + .send_keys() triggers unwanted behavior if a username is already set
......
...@@ -274,7 +274,7 @@ class CoursewareSearchCohortTest(ContainerBase): ...@@ -274,7 +274,7 @@ class CoursewareSearchCohortTest(ContainerBase):
Test staff user can search just student public content if selected from preview menu. Test staff user can search just student public content if selected from preview menu.
""" """
self._auto_auth(self.staff_user["username"], self.staff_user["email"], False) self._auto_auth(self.staff_user["username"], self.staff_user["email"], False)
self._goto_staff_page().set_staff_view_mode('Student') self._goto_staff_page().set_staff_view_mode('Learner')
self.courseware_search_page.search_for_term(self.visible_to_all_html) self.courseware_search_page.search_for_term(self.visible_to_all_html)
assert self.visible_to_all_html in self.courseware_search_page.search_results.html[0] assert self.visible_to_all_html in self.courseware_search_page.search_results.html[0]
self.courseware_search_page.clear_search() self.courseware_search_page.clear_search()
...@@ -292,7 +292,7 @@ class CoursewareSearchCohortTest(ContainerBase): ...@@ -292,7 +292,7 @@ class CoursewareSearchCohortTest(ContainerBase):
Test staff user can search cohort and public content if selected from preview menu. Test staff user can search cohort and public content if selected from preview menu.
""" """
self._auto_auth(self.staff_user["username"], self.staff_user["email"], False) self._auto_auth(self.staff_user["username"], self.staff_user["email"], False)
self._goto_staff_page().set_staff_view_mode('Student in ' + self.content_group_a) self._goto_staff_page().set_staff_view_mode('Learner in ' + self.content_group_a)
self.courseware_search_page.search_for_term(self.visible_to_all_html) self.courseware_search_page.search_for_term(self.visible_to_all_html)
assert self.visible_to_all_html in self.courseware_search_page.search_results.html[0] assert self.visible_to_all_html in self.courseware_search_page.search_results.html[0]
self.courseware_search_page.clear_search() self.courseware_search_page.clear_search()
......
...@@ -203,7 +203,7 @@ class GatingTest(UniqueCourseTest): ...@@ -203,7 +203,7 @@ class GatingTest(UniqueCourseTest):
self.courseware_page.wait_for_page() self.courseware_page.wait_for_page()
self.course_home_page.visit() self.course_home_page.visit()
self.course_home_page.preview.set_staff_view_mode('Student') self.course_home_page.preview.set_staff_view_mode('Learner')
self.assertEqual(self.course_home_page.outline.num_subsections, 1) self.assertEqual(self.course_home_page.outline.num_subsections, 1)
self.course_home_page.outline.go_to_section('Test Section 1', 'Test Subsection 1') self.course_home_page.outline.go_to_section('Test Section 1', 'Test Subsection 1')
self.courseware_page.wait_for_page() self.courseware_page.wait_for_page()
......
...@@ -302,7 +302,7 @@ class TestEntranceExamCourseImport(ImportTestMixin, StudioCourseTest): ...@@ -302,7 +302,7 @@ class TestEntranceExamCourseImport(ImportTestMixin, StudioCourseTest):
self.landing_page.view_live() self.landing_page.view_live()
courseware = CoursewarePage(self.browser, self.course_id) courseware = CoursewarePage(self.browser, self.course_id)
courseware.wait_for_page() courseware.wait_for_page()
StaffCoursewarePage(self.browser, self.course_id).set_staff_view_mode('Student') StaffCoursewarePage(self.browser, self.course_id).set_staff_view_mode('Learner')
self.assertEqual(courseware.num_sections, 1) self.assertEqual(courseware.num_sections, 1)
self.assertIn( self.assertIn(
"To access course materials, you must score", courseware.entrance_exam_message_selector.text[0] "To access course materials, you must score", courseware.entrance_exam_message_selector.text[0]
......
...@@ -1076,14 +1076,14 @@ class UnitPublishingTest(ContainerBase): ...@@ -1076,14 +1076,14 @@ class UnitPublishingTest(ContainerBase):
""" """
Verifies no component is visible when viewing as a student. Verifies no component is visible when viewing as a student.
""" """
self._verify_and_return_staff_page().set_staff_view_mode('Student') self._verify_and_return_staff_page().set_staff_view_mode('Learner')
self.assertEqual(0, self.courseware.num_xblock_components) self.assertEqual(0, self.courseware.num_xblock_components)
def _verify_student_view_visible(self, expected_components): def _verify_student_view_visible(self, expected_components):
""" """
Verifies expected components are visible when viewing as a student. Verifies expected components are visible when viewing as a student.
""" """
self._verify_and_return_staff_page().set_staff_view_mode('Student') self._verify_and_return_staff_page().set_staff_view_mode('Learner')
self._verify_components_visible(expected_components) self._verify_components_visible(expected_components)
def _verify_components_visible(self, expected_components): def _verify_components_visible(self, expected_components):
......
...@@ -743,7 +743,7 @@ class StaffLockTest(CourseOutlineTest): ...@@ -743,7 +743,7 @@ class StaffLockTest(CourseOutlineTest):
course_home_page = CourseHomePage(self.browser, self.course_id) course_home_page = CourseHomePage(self.browser, self.course_id)
course_home_page.visit() course_home_page.visit()
self.assertEqual(course_home_page.outline.num_sections, 2) self.assertEqual(course_home_page.outline.num_sections, 2)
course_home_page.preview.set_staff_view_mode('Student') course_home_page.preview.set_staff_view_mode('Learner')
self.assertEqual(course_home_page.outline.num_sections, 1) self.assertEqual(course_home_page.outline.num_sections, 1)
def test_locked_subsections_do_not_appear_in_lms(self): def test_locked_subsections_do_not_appear_in_lms(self):
...@@ -764,7 +764,7 @@ class StaffLockTest(CourseOutlineTest): ...@@ -764,7 +764,7 @@ class StaffLockTest(CourseOutlineTest):
course_home_page = CourseHomePage(self.browser, self.course_id) course_home_page = CourseHomePage(self.browser, self.course_id)
course_home_page.visit() course_home_page.visit()
self.assertEqual(course_home_page.outline.num_subsections, 2) self.assertEqual(course_home_page.outline.num_subsections, 2)
course_home_page.preview.set_staff_view_mode('Student') course_home_page.preview.set_staff_view_mode('Learner')
self.assertEqual(course_home_page.outline.num_subsections, 1) self.assertEqual(course_home_page.outline.num_subsections, 1)
def test_toggling_staff_lock_on_section_does_not_publish_draft_units(self): def test_toggling_staff_lock_on_section_does_not_publish_draft_units(self):
......
...@@ -17,6 +17,8 @@ from util.json_request import expect_json, JsonResponse ...@@ -17,6 +17,8 @@ from util.json_request import expect_json, JsonResponse
from opaque_keys.edx.keys import CourseKey from opaque_keys.edx.keys import CourseKey
from xblock.fragment import Fragment from xblock.fragment import Fragment
from xblock.runtime import KeyValueStore from xblock.runtime import KeyValueStore
from xmodule.partitions.partitions import NoSuchUserPartitionGroupError
log = logging.getLogger(__name__) log = logging.getLogger(__name__)
...@@ -170,16 +172,20 @@ def is_masquerading_as_specific_student(user, course_key): # pylint: disable=in ...@@ -170,16 +172,20 @@ def is_masquerading_as_specific_student(user, course_key): # pylint: disable=in
return bool(course_masquerade and course_masquerade.user_name) return bool(course_masquerade and course_masquerade.user_name)
def get_masquerading_group_info(user, course_key): def get_masquerading_user_group(course_key, user, user_partition):
""" """
If the user is masquerading as belonging to a group, then this method returns If the current user is masquerading as a generic learner in a specific group, return that group.
two values: the id of the group, and the id of the user partition that the group If the user is not masquerading as a group, then None is returned.
belongs to. If the user is not masquerading as a group, then None is returned.
""" """
course_masquerade = get_course_masquerade(user, course_key) course_masquerade = get_course_masquerade(user, course_key)
if not course_masquerade: if course_masquerade:
return None, None if course_masquerade.user_partition_id == user_partition.id and course_masquerade.group_id is not None:
return course_masquerade.group_id, course_masquerade.user_partition_id try:
return user_partition.get_group(course_masquerade.group_id)
except NoSuchUserPartitionGroupError:
return None
# The user is masquerading as a generic student or not masquerading as a group return None
return None
# Sentinel object to mark deleted objects in the session cache # Sentinel object to mark deleted objects in the session cache
......
...@@ -17,7 +17,7 @@ from courseware.masquerade import ( ...@@ -17,7 +17,7 @@ from courseware.masquerade import (
MasqueradingKeyValueStore, MasqueradingKeyValueStore,
handle_ajax, handle_ajax,
setup_masquerade, setup_masquerade,
get_masquerading_group_info get_masquerading_user_group
) )
from courseware.tests.factories import StaffFactory from courseware.tests.factories import StaffFactory
from courseware.tests.helpers import LoginEnrollmentTestCase, masquerade_as_group_member from courseware.tests.helpers import LoginEnrollmentTestCase, masquerade_as_group_member
...@@ -435,22 +435,20 @@ class TestGetMasqueradingGroupId(StaffMasqueradeTestCase): ...@@ -435,22 +435,20 @@ class TestGetMasqueradingGroupId(StaffMasqueradeTestCase):
modulestore().update_item(self.course, self.test_user.id) modulestore().update_item(self.course, self.test_user.id)
@patch.dict('django.conf.settings.FEATURES', {'DISABLE_START_DATES': False}) @patch.dict('django.conf.settings.FEATURES', {'DISABLE_START_DATES': False})
def test_group_masquerade(self): def test_get_masquerade_group(self):
""" """
Tests that a staff member can masquerade as being in a particular group. Tests that a staff member can masquerade as being in a group in a user partition
""" """
# Verify that there is no masquerading group initially # Verify there is no masquerading group initially
group_id, user_partition_id = get_masquerading_group_info(self.test_user, self.course.id) group = get_masquerading_user_group(self.course.id, self.test_user, self.user_partition)
self.assertIsNone(group_id) self.assertIsNone(group)
self.assertIsNone(user_partition_id)
# Install a masquerading group # Install a masquerading group
self.ensure_masquerade_as_group_member(0, 1) self.ensure_masquerade_as_group_member(0, 1)
# Verify that the masquerading group is returned # Verify that the masquerading group is returned
group_id, user_partition_id = get_masquerading_group_info(self.test_user, self.course.id) group = get_masquerading_user_group(self.course.id, self.test_user, self.user_partition)
self.assertEqual(group_id, 1) self.assertEqual(group.id, 1)
self.assertEqual(user_partition_id, 0)
class ReadOnlyKeyValueStore(DictKeyValueStore): class ReadOnlyKeyValueStore(DictKeyValueStore):
......
...@@ -6,10 +6,14 @@ ...@@ -6,10 +6,14 @@
<label for="action-preview-select" class="action-preview-label">View this course as:</label> <label for="action-preview-select" class="action-preview-label">View this course as:</label>
<select class="action-preview-select" id="action-preview-select" name="select"> <select class="action-preview-select" id="action-preview-select" name="select">
<option value="staff" selected>Staff</option> <option value="staff" selected>Staff</option>
<option value="student">Student</option> <option value="student">Learner</option>
<option value="specific student">Specific student</option> <option value="specific student">Specific learner</option>
<option value="group-a" data-group-id="group-a">Student in Group A</option> <option value="group-a" data-group-id="group-a" data-partition-id="test_partition_a_id">
<option value="group-b" data-group-id="group-b">Student in Group B</option> Learner in Group A
</option>
<option value="group-b" data-group-id="group-b" data-partition-id="test_partition_b_id">
Learner in Group B
</option>
</select> </select>
<div class="action-preview-username-container"> <div class="action-preview-username-container">
<label for="action-preview-username" class="action-preview-label">Username or email:</label> <label for="action-preview-username" class="action-preview-label">Username or email:</label>
......
...@@ -50,7 +50,7 @@ ...@@ -50,7 +50,7 @@
function masquerade(selectedOption) { function masquerade(selectedOption) {
var data = { var data = {
role: selectedOption.val() === 'staff' ? 'staff' : 'student', role: selectedOption.val() === 'staff' ? 'staff' : 'student',
user_partition_id: options.cohortedUserPartitionId, user_partition_id: selectedOption.data('partition-id'),
group_id: selectedOption.data('group-id'), group_id: selectedOption.data('group-id'),
user_name: selectedOption.val() === 'specific student' ? $userNameElement.val() : null user_name: selectedOption.val() === 'specific student' ? $userNameElement.val() : null
}; };
......
...@@ -70,7 +70,6 @@ define( ...@@ -70,7 +70,6 @@ define(
var requests = AjaxHelpers.requests(this), var requests = AjaxHelpers.requests(this),
reloadSpy = spyOn(ViewUtils, 'reload'); reloadSpy = spyOn(ViewUtils, 'reload');
showPreview({ showPreview({
cohortedUserPartitionId: 'test_partition_id',
courseId: 'test_course' courseId: 'test_course'
}); });
previewActionSelect().find('option[value="group-b"]').prop('selected', 'selected').change(); previewActionSelect().find('option[value="group-b"]').prop('selected', 'selected').change();
...@@ -79,7 +78,7 @@ define( ...@@ -79,7 +78,7 @@ define(
{ {
role: 'student', role: 'student',
user_name: null, user_name: null,
user_partition_id: 'test_partition_id', user_partition_id: 'test_partition_b_id',
group_id: 'group-b' group_id: 'group-b'
} }
); );
......
...@@ -5,9 +5,9 @@ ...@@ -5,9 +5,9 @@
<%! <%!
from django.utils.translation import ugettext as _ from django.utils.translation import ugettext as _
from django.conf import settings from django.conf import settings
from openedx.core.djangoapps.course_groups.partition_scheme import get_cohorted_user_partition
from openedx.core.djangolib.js_utils import dump_js_escaped_json from openedx.core.djangolib.js_utils import dump_js_escaped_json
from openedx.core.djangolib.markup import HTML, Text from openedx.core.djangolib.markup import HTML, Text
from xmodule.partitions.partitions_service import get_all_partitions_for_course
%> %>
<% <%
...@@ -19,7 +19,7 @@ show_preview_menu = course and staff_access and supports_preview_menu ...@@ -19,7 +19,7 @@ show_preview_menu = course and staff_access and supports_preview_menu
def selected(is_selected): def selected(is_selected):
return "selected" if is_selected else "" return "selected" if is_selected else ""
cohorted_user_partition = get_cohorted_user_partition(course) course_partitions = get_all_partitions_for_course(course)
masquerade_user_name = masquerade.user_name if masquerade else None masquerade_user_name = masquerade.user_name if masquerade else None
masquerade_group_id = masquerade.group_id if masquerade else None masquerade_group_id = masquerade.group_id if masquerade else None
staff_selected = selected(not masquerade or masquerade.role != "student") staff_selected = selected(not masquerade or masquerade.role != "student")
...@@ -34,13 +34,15 @@ show_preview_menu = course and staff_access and supports_preview_menu ...@@ -34,13 +34,15 @@ show_preview_menu = course and staff_access and supports_preview_menu
<label for="action-preview-select" class="action-preview-label">${_("View this course as:")}</label> <label for="action-preview-select" class="action-preview-label">${_("View this course as:")}</label>
<select class="action-preview-select" id="action-preview-select" name="select"> <select class="action-preview-select" id="action-preview-select" name="select">
<option value="staff" ${staff_selected}>${_("Staff")}</option> <option value="staff" ${staff_selected}>${_("Staff")}</option>
<option value="student" ${student_selected}>${_("Student")}</option> <option value="student" ${student_selected}>${_("Learner")}</option>
<option value="specific student" ${specific_student_selected}>${_("Specific student")}</option> <option value="specific student" ${specific_student_selected}>${_("Specific learner")}</option>
% if cohorted_user_partition: % if course_partitions:
% for group in sorted(cohorted_user_partition.groups, key=lambda group: group.name): % for course_partition in course_partitions:
<option value="group.id" data-group-id="${group.id}" ${selected(masquerade_group_id == group.id)}> % for group in sorted(course_partition.groups, key=lambda group: group.name):
${_("Student in {content_group}").format(content_group=group.name)} <option value="group.id" data-group-id="${group.id}" data-partition-id="${course_partition.id}" ${selected(masquerade_group_id == group.id)}>
</option> ${_("Learner in {content_group}").format(content_group=group.name)}
</option>
% endfor
% endfor % endfor
% endif % endif
</select> </select>
...@@ -71,7 +73,6 @@ show_preview_menu = course and staff_access and supports_preview_menu ...@@ -71,7 +73,6 @@ show_preview_menu = course and staff_access and supports_preview_menu
"courseId": course.id, "courseId": course.id,
"disableStudentAccess": disable_student_access if disable_student_access is not UNDEFINED else False, "disableStudentAccess": disable_student_access if disable_student_access is not UNDEFINED else False,
"specificStudentSelected": specific_student_selected, "specificStudentSelected": specific_student_selected,
"cohortedUserPartitionId": cohorted_user_partition.id if cohorted_user_partition else None,
"masqueradeUsername" : masquerade_user_name if masquerade_user_name is not UNDEFINED else None, "masqueradeUsername" : masquerade_user_name if masquerade_user_name is not UNDEFINED else None,
} }
%> %>
......
...@@ -5,8 +5,8 @@ import logging ...@@ -5,8 +5,8 @@ import logging
from courseware.masquerade import ( # pylint: disable=import-error from courseware.masquerade import ( # pylint: disable=import-error
get_course_masquerade, get_course_masquerade,
get_masquerading_group_info, get_masquerading_user_group,
is_masquerading_as_specific_student, is_masquerading_as_specific_student
) )
from xmodule.partitions.partitions import NoSuchUserPartitionGroupError from xmodule.partitions.partitions import NoSuchUserPartitionGroupError
...@@ -45,14 +45,7 @@ class CohortPartitionScheme(object): ...@@ -45,14 +45,7 @@ class CohortPartitionScheme(object):
# user is masquerading as a generic student in a specific group, then # user is masquerading as a generic student in a specific group, then
# return that group. # return that group.
if get_course_masquerade(user, course_key) and not is_masquerading_as_specific_student(user, course_key): if get_course_masquerade(user, course_key) and not is_masquerading_as_specific_student(user, course_key):
group_id, user_partition_id = get_masquerading_group_info(user, course_key) return get_masquerading_user_group(course_key, user, user_partition)
if user_partition_id == user_partition.id and group_id is not None:
try:
return user_partition.get_group(group_id)
except NoSuchUserPartitionGroupError:
return None
# The user is masquerading as a generic student. We can't show any particular group.
return None
cohort = get_cohort(user, course_key, use_cached=use_cached) cohort = get_cohort(user, course_key, use_cached=use_cached)
if cohort is None: if cohort is None:
......
...@@ -5,8 +5,8 @@ from django.conf import settings ...@@ -5,8 +5,8 @@ from django.conf import settings
from courseware.masquerade import ( from courseware.masquerade import (
get_course_masquerade, get_course_masquerade,
get_masquerading_group_info, get_masquerading_user_group,
is_masquerading_as_specific_student, is_masquerading_as_specific_student
) )
from course_modes.models import CourseMode from course_modes.models import CourseMode
from student.models import CourseEnrollment from student.models import CourseEnrollment
...@@ -79,24 +79,13 @@ class EnrollmentTrackPartitionScheme(object): ...@@ -79,24 +79,13 @@ class EnrollmentTrackPartitionScheme(object):
if is_course_using_cohort_instead(course_key): if is_course_using_cohort_instead(course_key):
return None return None
# NOTE: masquerade code was copied from CohortPartitionScheme, and it may need
# some changes (or if not, code should be refactored out and shared).
# This work will be done in a future story TNL-6739.
# First, check if we have to deal with masquerading. # First, check if we have to deal with masquerading.
# If the current user is masquerading as a specific student, use the # If the current user is masquerading as a specific student, use the
# same logic as normal to return that student's group. If the current # same logic as normal to return that student's group. If the current
# user is masquerading as a generic student in a specific group, then # user is masquerading as a generic student in a specific group, then
# return that group. # return that group.
if get_course_masquerade(user, course_key) and not is_masquerading_as_specific_student(user, course_key): if get_course_masquerade(user, course_key) and not is_masquerading_as_specific_student(user, course_key):
group_id, user_partition_id = get_masquerading_group_info(user, course_key) return get_masquerading_user_group(course_key, user, user_partition)
if user_partition_id == user_partition.id and group_id is not None:
try:
return user_partition.get_group(group_id)
except NoSuchUserPartitionGroupError:
return None
# The user is masquerading as a generic student. We can't show any particular group.
return None
mode_slug, is_active = CourseEnrollment.enrollment_mode_for_user(user, course_key) mode_slug, is_active = CourseEnrollment.enrollment_mode_for_user(user, course_key)
if mode_slug and is_active: if mode_slug and is_active:
......
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