This will remove imports from __future__ that are no longer needed. https://docs.python.org/3.5/library/2to3.html#2to3fixer-future
614 lines
22 KiB
Python
614 lines
22 KiB
Python
# -*- coding: utf-8 -*-
|
|
"""
|
|
End-to-end tests for the courseware unit bookmarks.
|
|
"""
|
|
|
|
|
|
import json
|
|
from unittest import skip
|
|
|
|
import requests
|
|
from six.moves import range
|
|
|
|
from common.test.acceptance.fixtures.course import CourseFixture, XBlockFixtureDesc
|
|
from common.test.acceptance.pages.common import BASE_URL
|
|
from common.test.acceptance.pages.common.auto_auth import AutoAuthPage
|
|
from common.test.acceptance.pages.common.logout import LogoutPage
|
|
from common.test.acceptance.pages.lms.bookmarks import BookmarksPage
|
|
from common.test.acceptance.pages.lms.course_home import CourseHomePage
|
|
from common.test.acceptance.pages.lms.courseware import CoursewarePage
|
|
from common.test.acceptance.pages.studio.overview import CourseOutlinePage as StudioCourseOutlinePage
|
|
from common.test.acceptance.tests.helpers import EventsTestMixin, UniqueCourseTest, is_404_page
|
|
|
|
|
|
class BookmarksTestMixin(EventsTestMixin, UniqueCourseTest):
|
|
"""
|
|
Mixin with helper methods for testing Bookmarks.
|
|
"""
|
|
USERNAME = "STUDENT"
|
|
EMAIL = "student@example.com"
|
|
|
|
def setUp(self):
|
|
super(BookmarksTestMixin, self).setUp()
|
|
|
|
self.studio_course_outline_page = StudioCourseOutlinePage(
|
|
self.browser,
|
|
self.course_info['org'],
|
|
self.course_info['number'],
|
|
self.course_info['run']
|
|
)
|
|
|
|
self.courseware_page = CoursewarePage(self.browser, self.course_id)
|
|
self.course_home_page = CourseHomePage(self.browser, self.course_id)
|
|
self.bookmarks_page = BookmarksPage(self.browser, self.course_id)
|
|
|
|
# Get session to be used for bookmarking units
|
|
self.session = requests.Session()
|
|
params = {'username': self.USERNAME, 'email': self.EMAIL, 'course_id': self.course_id}
|
|
response = self.session.get(BASE_URL + "/auto_auth", params=params)
|
|
self.assertTrue(response.ok, "Failed to get session")
|
|
|
|
def setup_test(self, num_chapters=2):
|
|
"""
|
|
Setup test settings.
|
|
|
|
Arguments:
|
|
num_chapters: number of chapters to create in course
|
|
"""
|
|
self.create_course_fixture(num_chapters)
|
|
|
|
# Auto-auth register for the course.
|
|
AutoAuthPage(self.browser, username=self.USERNAME, email=self.EMAIL, course_id=self.course_id).visit()
|
|
|
|
self.courseware_page.visit()
|
|
|
|
def create_course_fixture(self, num_chapters):
|
|
"""
|
|
Create course fixture
|
|
|
|
Arguments:
|
|
num_chapters: number of chapters to create
|
|
"""
|
|
self.course_fixture = CourseFixture(
|
|
self.course_info['org'], self.course_info['number'],
|
|
self.course_info['run'], self.course_info['display_name']
|
|
)
|
|
|
|
xblocks = []
|
|
for index in range(num_chapters):
|
|
xblocks += [
|
|
XBlockFixtureDesc('chapter', 'TestSection{}'.format(index)).add_children(
|
|
XBlockFixtureDesc('sequential', 'TestSubsection{}'.format(index)).add_children(
|
|
XBlockFixtureDesc('vertical', 'TestVertical{}'.format(index))
|
|
)
|
|
)
|
|
]
|
|
self.course_fixture.add_children(*xblocks).install()
|
|
|
|
def verify_event_data(self, event_type, event_data):
|
|
"""
|
|
Verify emitted event data.
|
|
|
|
Arguments:
|
|
event_type: expected event type
|
|
event_data: expected event data
|
|
"""
|
|
actual_events = self.wait_for_events(event_filter={'event_type': event_type}, number_of_matches=1)
|
|
self.assert_events_match(event_data, actual_events)
|
|
|
|
def _bookmark_unit(self, location):
|
|
"""
|
|
Bookmark a unit
|
|
|
|
Arguments:
|
|
location (str): unit location
|
|
"""
|
|
_headers = {
|
|
'Content-type': 'application/json',
|
|
'X-CSRFToken': self.session.cookies['csrftoken'],
|
|
}
|
|
params = {'course_id': self.course_id}
|
|
data = json.dumps({'usage_id': location})
|
|
response = self.session.post(
|
|
BASE_URL + '/api/bookmarks/v1/bookmarks/',
|
|
data=data,
|
|
params=params,
|
|
headers=_headers
|
|
)
|
|
self.assertTrue(response.ok, "Failed to bookmark unit")
|
|
|
|
def bookmark_units(self, num_units):
|
|
"""
|
|
Bookmark first `num_units` units
|
|
|
|
Arguments:
|
|
num_units(int): Number of units to bookmarks
|
|
"""
|
|
xblocks = self.course_fixture.get_nested_xblocks(category="vertical")
|
|
for index in range(num_units):
|
|
self._bookmark_unit(xblocks[index].locator)
|
|
|
|
|
|
class BookmarksTest(BookmarksTestMixin):
|
|
"""
|
|
Tests to verify bookmarks functionality.
|
|
"""
|
|
shard = 8
|
|
|
|
def _breadcrumb(self, num_units, modified_name=None):
|
|
"""
|
|
Creates breadcrumbs for the first `num_units`
|
|
|
|
Arguments:
|
|
num_units(int): Number of units for which we want to create breadcrumbs
|
|
|
|
Returns:
|
|
list of breadcrumbs
|
|
"""
|
|
breadcrumbs = []
|
|
for index in range(num_units):
|
|
breadcrumbs.append(
|
|
[
|
|
'TestSection{}'.format(index),
|
|
'TestSubsection{}'.format(index),
|
|
modified_name if modified_name else 'TestVertical{}'.format(index)
|
|
]
|
|
)
|
|
return breadcrumbs
|
|
|
|
def _delete_section(self, index):
|
|
""" Delete a section at index `index` """
|
|
|
|
# Logout and login as staff
|
|
LogoutPage(self.browser).visit()
|
|
AutoAuthPage(
|
|
self.browser, username=self.USERNAME, email=self.EMAIL, course_id=self.course_id, staff=True
|
|
).visit()
|
|
|
|
# Visit course outline page in studio.
|
|
self.studio_course_outline_page.visit()
|
|
self.studio_course_outline_page.wait_for_page()
|
|
|
|
self.studio_course_outline_page.section_at(index).delete()
|
|
|
|
# Logout and login as a student.
|
|
LogoutPage(self.browser).visit()
|
|
AutoAuthPage(self.browser, username=self.USERNAME, email=self.EMAIL, course_id=self.course_id).visit()
|
|
|
|
# Visit courseware as a student.
|
|
self.courseware_page.visit()
|
|
self.courseware_page.wait_for_page()
|
|
|
|
def _toggle_bookmark_and_verify(self, bookmark_icon_state, bookmark_button_state, bookmarked_count):
|
|
"""
|
|
Bookmark/Un-Bookmark a unit and then verify
|
|
"""
|
|
self.assertTrue(self.courseware_page.bookmark_button_visible)
|
|
self.courseware_page.click_bookmark_unit_button()
|
|
self.assertEqual(self.courseware_page.bookmark_icon_visible, bookmark_icon_state)
|
|
self.assertEqual(self.courseware_page.bookmark_button_state, bookmark_button_state)
|
|
self.bookmarks_page.visit()
|
|
self.assertEqual(self.bookmarks_page.count(), bookmarked_count)
|
|
|
|
def _verify_pagination_info(
|
|
self,
|
|
bookmark_count_on_current_page,
|
|
header_text,
|
|
previous_button_enabled,
|
|
next_button_enabled,
|
|
current_page_number,
|
|
total_pages
|
|
):
|
|
"""
|
|
Verify pagination info
|
|
"""
|
|
self.assertEqual(self.bookmarks_page.count(), bookmark_count_on_current_page)
|
|
self.assertEqual(self.bookmarks_page.get_pagination_header_text(), header_text)
|
|
self.assertEqual(self.bookmarks_page.is_previous_page_button_enabled(), previous_button_enabled)
|
|
self.assertEqual(self.bookmarks_page.is_next_page_button_enabled(), next_button_enabled)
|
|
self.assertEqual(self.bookmarks_page.get_current_page_number(), current_page_number)
|
|
self.assertEqual(self.bookmarks_page.get_total_pages, total_pages)
|
|
|
|
def _verify_breadcrumbs(self, num_units, modified_name=None):
|
|
"""
|
|
Verifies the breadcrumb trail.
|
|
"""
|
|
bookmarked_breadcrumbs = self.bookmarks_page.breadcrumbs()
|
|
|
|
# Verify bookmarked breadcrumbs.
|
|
breadcrumbs = self._breadcrumb(num_units=num_units, modified_name=modified_name)
|
|
breadcrumbs.reverse()
|
|
self.assertEqual(bookmarked_breadcrumbs, breadcrumbs)
|
|
|
|
def update_and_publish_block_display_name(self, modified_name):
|
|
"""
|
|
Update and publish the block/unit display name.
|
|
"""
|
|
self.studio_course_outline_page.visit()
|
|
self.studio_course_outline_page.wait_for_page()
|
|
|
|
self.studio_course_outline_page.expand_all_subsections()
|
|
section = self.studio_course_outline_page.section_at(0)
|
|
container_page = section.subsection_at(0).unit_at(0).go_to()
|
|
|
|
self.course_fixture._update_xblock(container_page.locator, { # pylint: disable=protected-access
|
|
"metadata": {
|
|
"display_name": modified_name
|
|
}
|
|
})
|
|
|
|
container_page.visit()
|
|
container_page.wait_for_page()
|
|
|
|
self.assertEqual(container_page.name, modified_name)
|
|
container_page.publish()
|
|
|
|
def test_bookmark_button(self):
|
|
"""
|
|
Scenario: Bookmark unit button toggles correctly
|
|
|
|
Given that I am a registered user
|
|
And I visit my courseware page
|
|
For first 2 units
|
|
I visit the unit
|
|
And I can see the Bookmark button
|
|
When I click on Bookmark button
|
|
Then unit should be bookmarked
|
|
Then I click again on the bookmark button
|
|
And I should see a unit un-bookmarked
|
|
"""
|
|
self.setup_test()
|
|
for index in range(2):
|
|
self.course_home_page.visit()
|
|
self.course_home_page.outline.go_to_section('TestSection{}'.format(index), 'TestSubsection{}'.format(index))
|
|
|
|
self._toggle_bookmark_and_verify(True, 'bookmarked', 1)
|
|
self.course_home_page.visit()
|
|
self.course_home_page.outline.go_to_section('TestSection{}'.format(index), 'TestSubsection{}'.format(index))
|
|
self._toggle_bookmark_and_verify(False, '', 0)
|
|
|
|
# TODO: TNL-6546: Remove this test
|
|
def test_courseware_bookmarks_button(self):
|
|
"""
|
|
Scenario: (Temporarily) test that the courseware's "Bookmarks" button works.
|
|
"""
|
|
self.setup_test()
|
|
self.bookmark_units(2)
|
|
self.courseware_page.visit()
|
|
self.courseware_page.click_bookmarks_button()
|
|
self.assertTrue(self.bookmarks_page.is_browser_on_page())
|
|
|
|
def test_empty_bookmarks_list(self):
|
|
"""
|
|
Scenario: An empty bookmarks list is shown if there are no bookmarked units.
|
|
|
|
Given that I am a registered user
|
|
And I visit my bookmarks page
|
|
Then I should see an empty bookmarks list
|
|
And empty bookmarks list content is correct
|
|
"""
|
|
self.setup_test()
|
|
self.bookmarks_page.visit()
|
|
empty_list_text = (
|
|
'Use bookmarks to help you easily return to courseware pages. '
|
|
'To bookmark a page, click "Bookmark this page" under the page title.')
|
|
self.assertEqual(self.bookmarks_page.empty_list_text(), empty_list_text)
|
|
|
|
def test_bookmarks_list(self):
|
|
"""
|
|
Scenario: A bookmarks list is shown if there are bookmarked units.
|
|
|
|
Given that I am a registered user
|
|
And I have bookmarked 2 units
|
|
And I visit my bookmarks page
|
|
Then I should see a bookmarked list with 2 bookmark links
|
|
And breadcrumb trail is correct for a bookmark
|
|
When I click on bookmarked link
|
|
Then I can navigate to correct bookmarked unit
|
|
"""
|
|
self.setup_test()
|
|
self.bookmark_units(2)
|
|
self.bookmarks_page.visit()
|
|
self._verify_breadcrumbs(num_units=2)
|
|
|
|
self._verify_pagination_info(
|
|
bookmark_count_on_current_page=2,
|
|
header_text='Showing 1-2 out of 2 total',
|
|
previous_button_enabled=False,
|
|
next_button_enabled=False,
|
|
current_page_number=1,
|
|
total_pages=1
|
|
)
|
|
|
|
# get usage ids for units
|
|
xblocks = self.course_fixture.get_nested_xblocks(category="vertical")
|
|
xblock_usage_ids = [xblock.locator for xblock in xblocks]
|
|
# Verify link navigation
|
|
for index in range(2):
|
|
self.bookmarks_page.visit()
|
|
self.bookmarks_page.click_bookmarked_block(index)
|
|
self.courseware_page.wait_for_page()
|
|
self.assertIn(self.courseware_page.active_usage_id(), xblock_usage_ids)
|
|
|
|
def test_bookmark_shows_updated_breadcrumb_after_publish(self):
|
|
"""
|
|
Scenario: A bookmark breadcrumb trail is updated after publishing the changed display name.
|
|
|
|
Given that I am a registered user
|
|
And I visit my courseware page
|
|
And I can see bookmarked unit
|
|
Then I visit unit page in studio
|
|
Then I change unit display_name
|
|
And I publish the changes
|
|
Then I visit my bookmarks page
|
|
When I see the bookmark
|
|
Then I can see the breadcrumb trail has the updated display_name.
|
|
"""
|
|
self.setup_test(num_chapters=1)
|
|
self.bookmark_units(num_units=1)
|
|
|
|
self.bookmarks_page.visit()
|
|
self._verify_breadcrumbs(num_units=1)
|
|
|
|
LogoutPage(self.browser).visit()
|
|
AutoAuthPage(
|
|
self.browser,
|
|
username=self.USERNAME,
|
|
email=self.EMAIL,
|
|
course_id=self.course_id,
|
|
staff=True
|
|
).visit()
|
|
|
|
modified_name = "Updated name"
|
|
self.update_and_publish_block_display_name(modified_name)
|
|
|
|
LogoutPage(self.browser).visit()
|
|
AutoAuthPage(self.browser, username=self.USERNAME, email=self.EMAIL, course_id=self.course_id).visit()
|
|
|
|
self.bookmarks_page.visit()
|
|
self._verify_breadcrumbs(num_units=1, modified_name=modified_name)
|
|
|
|
@skip("andya: 10/19/17: potentially flaky test")
|
|
def test_unreachable_bookmark(self):
|
|
"""
|
|
Scenario: We should get a HTTP 404 for an unreachable bookmark.
|
|
|
|
Given that I am a registered user
|
|
And I have bookmarked 2 units
|
|
And I delete a bookmarked unit
|
|
And I visit my bookmarks page
|
|
Then I should see a bookmarked list
|
|
When I click on the deleted bookmark
|
|
Then I should navigated to 404 page
|
|
"""
|
|
self.setup_test(num_chapters=1)
|
|
self.bookmark_units(1)
|
|
self._delete_section(0)
|
|
|
|
self.bookmarks_page.visit()
|
|
|
|
self._verify_pagination_info(
|
|
bookmark_count_on_current_page=1,
|
|
header_text='Showing 1 out of 1 total',
|
|
previous_button_enabled=False,
|
|
next_button_enabled=False,
|
|
current_page_number=1,
|
|
total_pages=1
|
|
)
|
|
|
|
self.bookmarks_page.click_bookmarked_block(0)
|
|
self.assertTrue(is_404_page(self.browser))
|
|
|
|
def test_page_size_limit(self):
|
|
"""
|
|
Scenario: We can't get bookmarks more than default page size.
|
|
|
|
Given that I am a registered user
|
|
And I have bookmarked all the 11 units available
|
|
And I visit my bookmarks page
|
|
Then I should see a bookmarked list
|
|
And the bookmark list should contain 10 bookmarked items
|
|
"""
|
|
self.setup_test(11)
|
|
self.bookmark_units(11)
|
|
self.bookmarks_page.visit()
|
|
|
|
self._verify_pagination_info(
|
|
bookmark_count_on_current_page=10,
|
|
header_text='Showing 1-10 out of 11 total',
|
|
previous_button_enabled=False,
|
|
next_button_enabled=True,
|
|
current_page_number=1,
|
|
total_pages=2
|
|
)
|
|
|
|
def test_pagination_with_single_page(self):
|
|
"""
|
|
Scenario: Bookmarks list pagination is working as expected for single page
|
|
Given that I am a registered user
|
|
And I have bookmarked all the 2 units available
|
|
And I visit my bookmarks page
|
|
Then I should see a bookmarked list with 2 bookmarked items
|
|
And I should see paging header and footer with correct data
|
|
And previous and next buttons are disabled
|
|
"""
|
|
self.setup_test(num_chapters=2)
|
|
self.bookmark_units(num_units=2)
|
|
self.bookmarks_page.visit()
|
|
self.assertTrue(self.bookmarks_page.results_present())
|
|
self._verify_pagination_info(
|
|
bookmark_count_on_current_page=2,
|
|
header_text='Showing 1-2 out of 2 total',
|
|
previous_button_enabled=False,
|
|
next_button_enabled=False,
|
|
current_page_number=1,
|
|
total_pages=1
|
|
)
|
|
|
|
def test_next_page_button(self):
|
|
"""
|
|
Scenario: Next button is working as expected for bookmarks list pagination
|
|
|
|
Given that I am a registered user
|
|
And I have bookmarked all the 12 units available
|
|
And I visit my bookmarks page
|
|
|
|
Then I should see a bookmarked list of 10 items
|
|
And I should see paging header and footer with correct info
|
|
|
|
Then I click on next page button in footer
|
|
And I should be navigated to second page
|
|
And I should see a bookmarked list with 2 items
|
|
And I should see paging header and footer with correct info
|
|
"""
|
|
self.setup_test(num_chapters=12)
|
|
self.bookmark_units(num_units=12)
|
|
|
|
self.bookmarks_page.visit()
|
|
self.assertTrue(self.bookmarks_page.results_present())
|
|
|
|
self._verify_pagination_info(
|
|
bookmark_count_on_current_page=10,
|
|
header_text='Showing 1-10 out of 12 total',
|
|
previous_button_enabled=False,
|
|
next_button_enabled=True,
|
|
current_page_number=1,
|
|
total_pages=2
|
|
)
|
|
|
|
self.bookmarks_page.press_next_page_button()
|
|
self._verify_pagination_info(
|
|
bookmark_count_on_current_page=2,
|
|
header_text='Showing 11-12 out of 12 total',
|
|
previous_button_enabled=True,
|
|
next_button_enabled=False,
|
|
current_page_number=2,
|
|
total_pages=2
|
|
)
|
|
|
|
def test_previous_page_button(self):
|
|
"""
|
|
Scenario: Previous button is working as expected for bookmarks list pagination
|
|
|
|
Given that I am a registered user
|
|
And I have bookmarked all the 12 units available
|
|
And I visit my bookmarks page
|
|
|
|
Then I click on next page button in footer
|
|
And I should be navigated to second page
|
|
And I should see a bookmarked list with 2 items
|
|
And I should see paging header and footer with correct info
|
|
|
|
Then I click on previous page button
|
|
And I should be navigated to first page
|
|
And I should see paging header and footer with correct info
|
|
"""
|
|
self.setup_test(num_chapters=12)
|
|
self.bookmark_units(num_units=12)
|
|
|
|
self.bookmarks_page.visit()
|
|
self.assertTrue(self.bookmarks_page.results_present())
|
|
|
|
self.bookmarks_page.press_next_page_button()
|
|
self._verify_pagination_info(
|
|
bookmark_count_on_current_page=2,
|
|
header_text='Showing 11-12 out of 12 total',
|
|
previous_button_enabled=True,
|
|
next_button_enabled=False,
|
|
current_page_number=2,
|
|
total_pages=2
|
|
)
|
|
|
|
self.bookmarks_page.press_previous_page_button()
|
|
self._verify_pagination_info(
|
|
bookmark_count_on_current_page=10,
|
|
header_text='Showing 1-10 out of 12 total',
|
|
previous_button_enabled=False,
|
|
next_button_enabled=True,
|
|
current_page_number=1,
|
|
total_pages=2
|
|
)
|
|
|
|
def test_pagination_with_valid_page_number(self):
|
|
"""
|
|
Scenario: Bookmarks list pagination works as expected for valid page number
|
|
|
|
Given that I am a registered user
|
|
And I have bookmarked all the 12 units available
|
|
And I visit my bookmarks page
|
|
Then I should see a bookmarked list
|
|
And I should see total page value is 2
|
|
Then I enter 2 in the page number input
|
|
And I should be navigated to page 2
|
|
"""
|
|
self.setup_test(num_chapters=11)
|
|
self.bookmark_units(num_units=11)
|
|
|
|
self.bookmarks_page.visit()
|
|
self.assertTrue(self.bookmarks_page.results_present())
|
|
self.assertEqual(self.bookmarks_page.get_total_pages, 2)
|
|
|
|
self.bookmarks_page.go_to_page(2)
|
|
self._verify_pagination_info(
|
|
bookmark_count_on_current_page=1,
|
|
header_text='Showing 11-11 out of 11 total',
|
|
previous_button_enabled=True,
|
|
next_button_enabled=False,
|
|
current_page_number=2,
|
|
total_pages=2
|
|
)
|
|
|
|
def test_pagination_with_invalid_page_number(self):
|
|
"""
|
|
Scenario: Bookmarks list pagination works as expected for invalid page number
|
|
|
|
Given that I am a registered user
|
|
And I have bookmarked all the 11 units available
|
|
And I visit my bookmarks page
|
|
Then I should see a bookmarked list
|
|
And I should see total page value is 2
|
|
Then I enter 3 in the page number input
|
|
And I should stay at page 1
|
|
"""
|
|
self.setup_test(num_chapters=11)
|
|
self.bookmark_units(num_units=11)
|
|
|
|
self.bookmarks_page.visit()
|
|
self.assertTrue(self.bookmarks_page.results_present())
|
|
self.assertEqual(self.bookmarks_page.get_total_pages, 2)
|
|
|
|
self.bookmarks_page.go_to_page(3)
|
|
self._verify_pagination_info(
|
|
bookmark_count_on_current_page=10,
|
|
header_text='Showing 1-10 out of 11 total',
|
|
previous_button_enabled=False,
|
|
next_button_enabled=True,
|
|
current_page_number=1,
|
|
total_pages=2
|
|
)
|
|
|
|
|
|
class BookmarksA11yTests(BookmarksTestMixin):
|
|
"""
|
|
Tests for checking the a11y of the bookmarks page.
|
|
"""
|
|
a11y = True
|
|
|
|
def test_view_a11y(self):
|
|
"""
|
|
Verify the basic accessibility of the bookmarks page while paginated.
|
|
"""
|
|
self.bookmarks_page.a11y_audit.config.set_rules({
|
|
"ignore": [
|
|
'aria-valid-attr', # TODO: LEARNER-6611 & LEARNER-6865
|
|
'heading-order', # TODO: AC-933
|
|
'landmark-no-duplicate-banner', # TODO: AC-934
|
|
'landmark-one-main', # TODO: AC-944
|
|
'region' # TODO: AC-932
|
|
]
|
|
})
|
|
self.setup_test(num_chapters=11)
|
|
self.bookmark_units(num_units=11)
|
|
self.bookmarks_page.visit()
|
|
self.bookmarks_page.a11y_audit.check_for_accessibility_errors()
|