Commit a70a23a4 by Jay Zoldak

Merge pull request #1680 from MITx/feature/will/courseware_lettuce_tests

Feature/will/courseware lettuce tests
parents c50bf567 568f557d
...@@ -3,6 +3,11 @@ from splinter.browser import Browser ...@@ -3,6 +3,11 @@ from splinter.browser import Browser
from logging import getLogger from logging import getLogger
import time import time
# Let the LMS and CMS do their one-time setup
# For example, setting up mongo caches
from lms import one_time_startup
from cms import one_time_startup
logger = getLogger(__name__) logger = getLogger(__name__)
logger.info("Loading the lettuce acceptance testing terrain file...") logger.info("Loading the lettuce acceptance testing terrain file...")
......
...@@ -121,21 +121,41 @@ class XModuleItemFactory(Factory): ...@@ -121,21 +121,41 @@ class XModuleItemFactory(Factory):
@classmethod @classmethod
def _create(cls, target_class, *args, **kwargs): def _create(cls, target_class, *args, **kwargs):
""" """
kwargs must include parent_location, template. Can contain display_name Uses *kwargs*:
target_class is ignored
*parent_location* (required): the location of the parent module
(e.g. the parent course or section)
*template* (required): the template to create the item from
(e.g. i4x://templates/section/Empty)
*data* (optional): the data for the item
(e.g. XML problem definition for a problem item)
*display_name* (optional): the display name of the item
*metadata* (optional): dictionary of metadata attributes
*target_class* is ignored
""" """
DETACHED_CATEGORIES = ['about', 'static_tab', 'course_info'] DETACHED_CATEGORIES = ['about', 'static_tab', 'course_info']
parent_location = Location(kwargs.get('parent_location')) parent_location = Location(kwargs.get('parent_location'))
template = Location(kwargs.get('template')) template = Location(kwargs.get('template'))
data = kwargs.get('data')
display_name = kwargs.get('display_name') display_name = kwargs.get('display_name')
metadata = kwargs.get('metadata', {})
store = modulestore('direct') store = modulestore('direct')
# This code was based off that in cms/djangoapps/contentstore/views.py # This code was based off that in cms/djangoapps/contentstore/views.py
parent = store.get_item(parent_location) parent = store.get_item(parent_location)
dest_location = parent_location._replace(category=template.category, name=uuid4().hex)
# If a display name is set, use that
dest_name = display_name.replace(" ", "_") if display_name is not None else uuid4().hex
dest_location = parent_location._replace(category=template.category,
name=dest_name)
new_item = store.clone_item(template, dest_location) new_item = store.clone_item(template, dest_location)
...@@ -143,7 +163,14 @@ class XModuleItemFactory(Factory): ...@@ -143,7 +163,14 @@ class XModuleItemFactory(Factory):
if display_name is not None: if display_name is not None:
new_item.display_name = display_name new_item.display_name = display_name
store.update_metadata(new_item.location.url(), own_metadata(new_item)) # Add additional metadata or override current metadata
item_metadata = own_metadata(new_item)
item_metadata.update(metadata)
store.update_metadata(new_item.location.url(), item_metadata)
# replace the data with the optional *data* parameter
if data is not None:
store.update_item(new_item.location, data)
if new_item.location.category not in DETACHED_CATEGORIES: if new_item.location.category not in DETACHED_CATEGORIES:
store.update_children(parent_location, parent.children + [new_item.location.url()]) store.update_children(parent_location, parent.children + [new_item.location.url()])
......
...@@ -69,6 +69,11 @@ def the_page_title_should_be(step, title): ...@@ -69,6 +69,11 @@ def the_page_title_should_be(step, title):
assert_equals(world.browser.title, title) assert_equals(world.browser.title, title)
@step(u'the page title should contain "([^"]*)"$')
def the_page_title_should_contain(step, title):
assert(title in world.browser.title)
@step('I am a logged in user$') @step('I am a logged in user$')
def i_am_logged_in_user(step): def i_am_logged_in_user(step):
create_user('robot') create_user('robot')
...@@ -80,18 +85,6 @@ def i_am_not_logged_in(step): ...@@ -80,18 +85,6 @@ def i_am_not_logged_in(step):
world.browser.cookies.delete() world.browser.cookies.delete()
@step('I am registered for a course$')
def i_am_registered_for_a_course(step):
create_user('robot')
u = User.objects.get(username='robot')
CourseEnrollment.objects.get_or_create(user=u, course_id='MITx/6.002x/2012_Fall')
@step('I am registered for course "([^"]*)"$')
def i_am_registered_for_course_by_id(step, course_id):
register_by_course_id(course_id)
@step('I am staff for course "([^"]*)"$') @step('I am staff for course "([^"]*)"$')
def i_am_staff_for_course_by_id(step, course_id): def i_am_staff_for_course_by_id(step, course_id):
register_by_course_id(course_id, True) register_by_course_id(course_id, True)
...@@ -108,6 +101,7 @@ def i_am_an_edx_user(step): ...@@ -108,6 +101,7 @@ def i_am_an_edx_user(step):
#### helper functions #### helper functions
@world.absorb @world.absorb
def scroll_to_bottom(): def scroll_to_bottom():
# Maximize the browser # Maximize the browser
...@@ -116,6 +110,11 @@ def scroll_to_bottom(): ...@@ -116,6 +110,11 @@ def scroll_to_bottom():
@world.absorb @world.absorb
def create_user(uname): def create_user(uname):
# If the user already exists, don't try to create it again
if len(User.objects.filter(username=uname)) > 0:
return
portal_user = UserFactory.build(username=uname, email=uname + '@edx.org') portal_user = UserFactory.build(username=uname, email=uname + '@edx.org')
portal_user.set_password('test') portal_user.set_password('test')
portal_user.save() portal_user.save()
...@@ -133,13 +132,25 @@ def log_in(email, password): ...@@ -133,13 +132,25 @@ def log_in(email, password):
world.browser.visit(django_url('/')) world.browser.visit(django_url('/'))
world.browser.is_element_present_by_css('header.global', 10) world.browser.is_element_present_by_css('header.global', 10)
world.browser.click_link_by_href('#login-modal') world.browser.click_link_by_href('#login-modal')
login_form = world.browser.find_by_css('form#login_form')
# Wait for the login dialog to load
# This is complicated by the fact that sometimes a second #login_form
# dialog loads, while the first one remains hidden.
# We give them both time to load, starting with the second one.
world.browser.is_element_present_by_css('section.content-wrapper form#login_form', wait_time=4)
world.browser.is_element_present_by_css('form#login_form', wait_time=2)
# For some reason, the page sometimes includes two #login_form
# elements, the first of which is not visible.
# To avoid this, we always select the last of the two #login_form dialogs
login_form = world.browser.find_by_css('form#login_form').last
login_form.find_by_name('email').fill(email) login_form.find_by_name('email').fill(email)
login_form.find_by_name('password').fill(password) login_form.find_by_name('password').fill(password)
login_form.find_by_name('submit').click() login_form.find_by_name('submit').click()
# wait for the page to redraw # wait for the page to redraw
assert world.browser.is_element_present_by_css('.content-wrapper', 10) assert world.browser.is_element_present_by_css('.content-wrapper', wait_time=10)
@world.absorb @world.absorb
......
from lxml import etree from lxml import etree
from abc import ABCMeta, abstractmethod from abc import ABCMeta, abstractmethod
class ResponseXMLFactory(object): class ResponseXMLFactory(object):
""" Abstract base class for capa response XML factories. """ Abstract base class for capa response XML factories.
Subclasses override create_response_element and Subclasses override create_response_element and
...@@ -135,7 +136,7 @@ class ResponseXMLFactory(object): ...@@ -135,7 +136,7 @@ class ResponseXMLFactory(object):
# Names of group elements # Names of group elements
group_element_names = {'checkbox': 'checkboxgroup', group_element_names = {'checkbox': 'checkboxgroup',
'radio': 'radiogroup', 'radio': 'radiogroup',
'multiple': 'choicegroup' } 'multiple': 'choicegroup'}
# Retrieve **kwargs # Retrieve **kwargs
choices = kwargs.get('choices', [True]) choices = kwargs.get('choices', [True])
...@@ -151,13 +152,11 @@ class ResponseXMLFactory(object): ...@@ -151,13 +152,11 @@ class ResponseXMLFactory(object):
choice_element = etree.SubElement(group_element, "choice") choice_element = etree.SubElement(group_element, "choice")
choice_element.set("correct", "true" if correct_val else "false") choice_element.set("correct", "true" if correct_val else "false")
# Add some text describing the choice
etree.SubElement(choice_element, "startouttext")
etree.text = "Choice description"
etree.SubElement(choice_element, "endouttext")
# Add a name identifying the choice, if one exists # Add a name identifying the choice, if one exists
# For simplicity, we use the same string as both the
# name attribute and the text of the element
if name: if name:
choice_element.text = str(name)
choice_element.set("name", str(name)) choice_element.set("name", str(name))
return group_element return group_element
...@@ -274,6 +273,7 @@ class SchematicResponseXMLFactory(ResponseXMLFactory): ...@@ -274,6 +273,7 @@ class SchematicResponseXMLFactory(ResponseXMLFactory):
For testing, we create a bare-bones version of <schematic>.""" For testing, we create a bare-bones version of <schematic>."""
return etree.Element("schematic") return etree.Element("schematic")
class CodeResponseXMLFactory(ResponseXMLFactory): class CodeResponseXMLFactory(ResponseXMLFactory):
""" Factory for creating <coderesponse> XML trees """ """ Factory for creating <coderesponse> XML trees """
...@@ -328,6 +328,7 @@ class CodeResponseXMLFactory(ResponseXMLFactory): ...@@ -328,6 +328,7 @@ class CodeResponseXMLFactory(ResponseXMLFactory):
# return None here # return None here
return None return None
class ChoiceResponseXMLFactory(ResponseXMLFactory): class ChoiceResponseXMLFactory(ResponseXMLFactory):
""" Factory for creating <choiceresponse> XML trees """ """ Factory for creating <choiceresponse> XML trees """
...@@ -440,6 +441,7 @@ class FormulaResponseXMLFactory(ResponseXMLFactory): ...@@ -440,6 +441,7 @@ class FormulaResponseXMLFactory(ResponseXMLFactory):
"#" + str(num_samples)) "#" + str(num_samples))
return sample_str return sample_str
class ImageResponseXMLFactory(ResponseXMLFactory): class ImageResponseXMLFactory(ResponseXMLFactory):
""" Factory for producing <imageresponse> XML """ """ Factory for producing <imageresponse> XML """
...@@ -499,6 +501,7 @@ class ImageResponseXMLFactory(ResponseXMLFactory): ...@@ -499,6 +501,7 @@ class ImageResponseXMLFactory(ResponseXMLFactory):
return input_element return input_element
class JavascriptResponseXMLFactory(ResponseXMLFactory): class JavascriptResponseXMLFactory(ResponseXMLFactory):
""" Factory for producing <javascriptresponse> XML """ """ Factory for producing <javascriptresponse> XML """
...@@ -552,6 +555,7 @@ class JavascriptResponseXMLFactory(ResponseXMLFactory): ...@@ -552,6 +555,7 @@ class JavascriptResponseXMLFactory(ResponseXMLFactory):
""" Create the <javascriptinput> element """ """ Create the <javascriptinput> element """
return etree.Element("javascriptinput") return etree.Element("javascriptinput")
class MultipleChoiceResponseXMLFactory(ResponseXMLFactory): class MultipleChoiceResponseXMLFactory(ResponseXMLFactory):
""" Factory for producing <multiplechoiceresponse> XML """ """ Factory for producing <multiplechoiceresponse> XML """
...@@ -564,6 +568,7 @@ class MultipleChoiceResponseXMLFactory(ResponseXMLFactory): ...@@ -564,6 +568,7 @@ class MultipleChoiceResponseXMLFactory(ResponseXMLFactory):
kwargs['choice_type'] = 'multiple' kwargs['choice_type'] = 'multiple'
return ResponseXMLFactory.choicegroup_input_xml(**kwargs) return ResponseXMLFactory.choicegroup_input_xml(**kwargs)
class TrueFalseResponseXMLFactory(ResponseXMLFactory): class TrueFalseResponseXMLFactory(ResponseXMLFactory):
""" Factory for producing <truefalseresponse> XML """ """ Factory for producing <truefalseresponse> XML """
...@@ -576,6 +581,7 @@ class TrueFalseResponseXMLFactory(ResponseXMLFactory): ...@@ -576,6 +581,7 @@ class TrueFalseResponseXMLFactory(ResponseXMLFactory):
kwargs['choice_type'] = 'multiple' kwargs['choice_type'] = 'multiple'
return ResponseXMLFactory.choicegroup_input_xml(**kwargs) return ResponseXMLFactory.choicegroup_input_xml(**kwargs)
class OptionResponseXMLFactory(ResponseXMLFactory): class OptionResponseXMLFactory(ResponseXMLFactory):
""" Factory for producing <optionresponse> XML""" """ Factory for producing <optionresponse> XML"""
...@@ -667,6 +673,7 @@ class StringResponseXMLFactory(ResponseXMLFactory): ...@@ -667,6 +673,7 @@ class StringResponseXMLFactory(ResponseXMLFactory):
def create_input_element(self, **kwargs): def create_input_element(self, **kwargs):
return ResponseXMLFactory.textline_input_xml(**kwargs) return ResponseXMLFactory.textline_input_xml(**kwargs)
class AnnotationResponseXMLFactory(ResponseXMLFactory): class AnnotationResponseXMLFactory(ResponseXMLFactory):
""" Factory for creating <annotationresponse> XML trees """ """ Factory for creating <annotationresponse> XML trees """
def create_response_element(self, **kwargs): def create_response_element(self, **kwargs):
...@@ -679,17 +686,17 @@ class AnnotationResponseXMLFactory(ResponseXMLFactory): ...@@ -679,17 +686,17 @@ class AnnotationResponseXMLFactory(ResponseXMLFactory):
input_element = etree.Element("annotationinput") input_element = etree.Element("annotationinput")
text_children = [ text_children = [
{'tag': 'title', 'text': kwargs.get('title', 'super cool annotation') }, {'tag': 'title', 'text': kwargs.get('title', 'super cool annotation')},
{'tag': 'text', 'text': kwargs.get('text', 'texty text') }, {'tag': 'text', 'text': kwargs.get('text', 'texty text')},
{'tag': 'comment', 'text':kwargs.get('comment', 'blah blah erudite comment blah blah') }, {'tag': 'comment', 'text':kwargs.get('comment', 'blah blah erudite comment blah blah')},
{'tag': 'comment_prompt', 'text': kwargs.get('comment_prompt', 'type a commentary below') }, {'tag': 'comment_prompt', 'text': kwargs.get('comment_prompt', 'type a commentary below')},
{'tag': 'tag_prompt', 'text': kwargs.get('tag_prompt', 'select one tag') } {'tag': 'tag_prompt', 'text': kwargs.get('tag_prompt', 'select one tag')}
] ]
for child in text_children: for child in text_children:
etree.SubElement(input_element, child['tag']).text = child['text'] etree.SubElement(input_element, child['tag']).text = child['text']
default_options = [('green', 'correct'),('eggs', 'incorrect'),('ham', 'partially-correct')] default_options = [('green', 'correct'),('eggs', 'incorrect'), ('ham', 'partially-correct')]
options = kwargs.get('options', default_options) options = kwargs.get('options', default_options)
options_element = etree.SubElement(input_element, 'options') options_element = etree.SubElement(input_element, 'options')
...@@ -698,4 +705,3 @@ class AnnotationResponseXMLFactory(ResponseXMLFactory): ...@@ -698,4 +705,3 @@ class AnnotationResponseXMLFactory(ResponseXMLFactory):
option_element.text = description option_element.text = description
return input_element return input_element
...@@ -5,6 +5,10 @@ from lettuce.django import django_url ...@@ -5,6 +5,10 @@ from lettuce.django import django_url
from django.conf import settings from django.conf import settings
from django.contrib.auth.models import User from django.contrib.auth.models import User
from student.models import CourseEnrollment from student.models import CourseEnrollment
from terrain.factories import CourseFactory, ItemFactory
from xmodule.modulestore import Location
from xmodule.modulestore.django import _MODULESTORES, modulestore
from xmodule.templates import update_templates
import time import time
from logging import getLogger from logging import getLogger
...@@ -81,14 +85,57 @@ def i_am_not_logged_in(step): ...@@ -81,14 +85,57 @@ def i_am_not_logged_in(step):
world.browser.cookies.delete() world.browser.cookies.delete()
@step(u'I am registered for a course$') TEST_COURSE_ORG = 'edx'
def i_am_registered_for_a_course(step): TEST_COURSE_NAME = 'Test Course'
TEST_SECTION_NAME = "Problem"
@step(u'The course "([^"]*)" exists$')
def create_course(step, course):
# First clear the modulestore so we don't try to recreate
# the same course twice
# This also ensures that the necessary templates are loaded
flush_xmodule_store()
# Create the course
# We always use the same org and display name,
# but vary the course identifier (e.g. 600x or 191x)
course = CourseFactory.create(org=TEST_COURSE_ORG,
number=course,
display_name=TEST_COURSE_NAME)
# Add a section to the course to contain problems
section = ItemFactory.create(parent_location=course.location,
display_name=TEST_SECTION_NAME)
problem_section = ItemFactory.create(parent_location=section.location,
template='i4x://edx/templates/sequential/Empty',
display_name=TEST_SECTION_NAME)
@step(u'I am registered for the course "([^"]*)"$')
def i_am_registered_for_the_course(step, course):
# Create the course
create_course(step, course)
# Create the user
world.create_user('robot') world.create_user('robot')
u = User.objects.get(username='robot') u = User.objects.get(username='robot')
CourseEnrollment.objects.create(user=u, course_id='MITx/6.002x/2012_Fall')
# If the user is not already enrolled, enroll the user.
CourseEnrollment.objects.get_or_create(user=u, course_id=course_id(course))
world.log_in('robot@edx.org', 'test') world.log_in('robot@edx.org', 'test')
@step(u'The course "([^"]*)" has extra tab "([^"]*)"$')
def add_tab_to_course(step, course, extra_tab_name):
section_item = ItemFactory.create(parent_location=course_location(course),
template="i4x://edx/templates/static_tab/Empty",
display_name=str(extra_tab_name))
@step(u'I am an edX user$') @step(u'I am an edX user$')
def i_am_an_edx_user(step): def i_am_an_edx_user(step):
world.create_user('robot') world.create_user('robot')
...@@ -97,3 +144,37 @@ def i_am_an_edx_user(step): ...@@ -97,3 +144,37 @@ def i_am_an_edx_user(step):
@step(u'User "([^"]*)" is an edX user$') @step(u'User "([^"]*)" is an edX user$')
def registered_edx_user(step, uname): def registered_edx_user(step, uname):
world.create_user(uname) world.create_user(uname)
def flush_xmodule_store():
# Flush and initialize the module store
# It needs the templates because it creates new records
# by cloning from the template.
# Note that if your test module gets in some weird state
# (though it shouldn't), do this manually
# from the bash shell to drop it:
# $ mongo test_xmodule --eval "db.dropDatabase()"
_MODULESTORES = {}
modulestore().collection.drop()
update_templates()
def course_id(course_num):
return "%s/%s/%s" % (TEST_COURSE_ORG, course_num,
TEST_COURSE_NAME.replace(" ", "_"))
def course_location(course_num):
return Location(loc_or_tag="i4x",
org=TEST_COURSE_ORG,
course=course_num,
category='course',
name=TEST_COURSE_NAME.replace(" ", "_"))
def section_location(course_num):
return Location(loc_or_tag="i4x",
org=TEST_COURSE_ORG,
course=course_num,
category='sequential',
name=TEST_SECTION_NAME.replace(" ", "_"))
...@@ -9,6 +9,7 @@ logger = getLogger(__name__) ...@@ -9,6 +9,7 @@ logger = getLogger(__name__)
## support functions ## support functions
def get_courses(): def get_courses():
''' '''
Returns dict of lists of courses available, keyed by course.org (ie university). Returns dict of lists of courses available, keyed by course.org (ie university).
......
Feature: View the Courseware Tab
As a student in an edX course
In order to work on the course
I want to view the info on the courseware tab
Scenario: I can get to the courseware tab when logged in
Given I am registered for a course
And I log in
And I click on View Courseware
When I click on the "Courseware" tab
Then the "Courseware" tab is active
...@@ -3,21 +3,18 @@ Feature: All the high level tabs should work ...@@ -3,21 +3,18 @@ Feature: All the high level tabs should work
As a student As a student
I want to navigate through the high level tabs I want to navigate through the high level tabs
# Note this didn't work as a scenario outline because Scenario: I can navigate to all high -level tabs in a course
# before each scenario was not flushing the database Given: I am registered for the course "6.002x"
# TODO: break this apart so that if one fails the others And The course "6.002x" has extra tab "Custom Tab"
# will still run
Scenario: A student can see all tabs of the course
Given I am registered for a course
And I log in And I log in
And I click on View Courseware And I click on View Courseware
When I click on the "Courseware" tab When I click on the "<TabName>" tab
Then the page title should be "6.002x Courseware" Then the page title should contain "<PageTitle>"
When I click on the "Course Info" tab
Then the page title should be "6.002x Course Info" Examples:
When I click on the "Textbook" tab | TabName | PageTitle |
Then the page title should be "6.002x Textbook" | Courseware | 6.002x Courseware |
When I click on the "Wiki" tab | Course Info | 6.002x Course Info |
Then the page title should be "6.002x | edX Wiki" | Custom Tab | 6.002x Custom Tab |
When I click on the "Progress" tab | Wiki | edX Wiki |
Then the page title should be "6.002x Progress" | Progress | 6.002x Progress |
...@@ -34,6 +34,7 @@ def click_the_dropdown(step): ...@@ -34,6 +34,7 @@ def click_the_dropdown(step):
#### helper functions #### helper functions
def user_is_an_unactivated_user(uname): def user_is_an_unactivated_user(uname):
u = User.objects.get(username=uname) u = User.objects.get(username=uname)
u.is_active = False u.is_active = False
......
Feature: Answer choice problems
As a student in an edX course
In order to test my understanding of the material
I want to answer choice based problems
Scenario: I can answer a problem correctly
Given I am viewing a "<ProblemType>" problem
When I answer a "<ProblemType>" problem "correctly"
Then My "<ProblemType>" answer is marked "correct"
Examples:
| ProblemType |
| drop down |
| multiple choice |
| checkbox |
| string |
| numerical |
| formula |
| script |
Scenario: I can answer a problem incorrectly
Given I am viewing a "<ProblemType>" problem
When I answer a "<ProblemType>" problem "incorrectly"
Then My "<ProblemType>" answer is marked "incorrect"
Examples:
| ProblemType |
| drop down |
| multiple choice |
| checkbox |
| string |
| numerical |
| formula |
| script |
Scenario: I can submit a blank answer
Given I am viewing a "<ProblemType>" problem
When I check a problem
Then My "<ProblemType>" answer is marked "incorrect"
Examples:
| ProblemType |
| drop down |
| multiple choice |
| checkbox |
| string |
| numerical |
| formula |
| script |
Scenario: I can reset a problem
Given I am viewing a "<ProblemType>" problem
And I answer a "<ProblemType>" problem "<Correctness>ly"
When I reset the problem
Then My "<ProblemType>" answer is marked "unanswered"
Examples:
| ProblemType | Correctness |
| drop down | correct |
| drop down | incorrect |
| multiple choice | correct |
| multiple choice | incorrect |
| checkbox | correct |
| checkbox | incorrect |
| string | correct |
| string | incorrect |
| numerical | correct |
| numerical | incorrect |
| formula | correct |
| formula | incorrect |
| script | correct |
| script | incorrect |
...@@ -4,13 +4,14 @@ Feature: Register for a course ...@@ -4,13 +4,14 @@ Feature: Register for a course
I want to register for a class on the edX website I want to register for a class on the edX website
Scenario: I can register for a course Scenario: I can register for a course
Given I am logged in Given The course "6.002x" exists
And I am logged in
And I visit the courses page And I visit the courses page
When I register for the course numbered "6.002x" When I register for the course "6.002x"
Then I should see the course numbered "6.002x" in my dashboard Then I should see the course numbered "6.002x" in my dashboard
Scenario: I can unregister for a course Scenario: I can unregister for a course
Given I am registered for a course Given I am registered for the course "6.002x"
And I visit the dashboard And I visit the dashboard
When I click the link with the text "Unregister" When I click the link with the text "Unregister"
And I press the "Unregister" button in the Unenroll dialog And I press the "Unregister" button in the Unenroll dialog
......
from lettuce import world, step from lettuce import world, step
from lettuce.django import django_url
from common import TEST_COURSE_ORG, TEST_COURSE_NAME
@step('I register for the course numbered "([^"]*)"$') @step('I register for the course "([^"]*)"$')
def i_register_for_the_course(step, course): def i_register_for_the_course(step, course):
courses_section = world.browser.find_by_css('section.courses') cleaned_name = TEST_COURSE_NAME.replace(' ', '_')
course_link_css = 'article[id*="%s"] > div' % course url = django_url('courses/%s/%s/%s/about' % (TEST_COURSE_ORG, course, cleaned_name))
course_link = courses_section.find_by_css(course_link_css).first world.browser.visit(url)
course_link.click()
intro_section = world.browser.find_by_css('section.intro') intro_section = world.browser.find_by_css('section.intro')
register_link = intro_section.find_by_css('a.register') register_link = intro_section.find_by_css('a.register')
......
...@@ -8,16 +8,24 @@ from .test import * ...@@ -8,16 +8,24 @@ from .test import *
# otherwise the browser will not render the pages correctly # otherwise the browser will not render the pages correctly
DEBUG = True DEBUG = True
# Show the courses that are in the data directory # Use the mongo store for acceptance tests
COURSES_ROOT = ENV_ROOT / "data" modulestore_options = {
DATA_DIR = COURSES_ROOT 'default_class': 'xmodule.raw_module.RawDescriptor',
'host': 'localhost',
'db': 'test_xmodule',
'collection': 'modulestore',
'fs_root': GITHUB_REPO_ROOT,
'render_template': 'mitxmako.shortcuts.render_to_string',
}
MODULESTORE = { MODULESTORE = {
'default': { 'default': {
'ENGINE': 'xmodule.modulestore.xml.XMLModuleStore', 'ENGINE': 'xmodule.modulestore.mongo.MongoModuleStore',
'OPTIONS': { 'OPTIONS': modulestore_options
'data_dir': DATA_DIR, },
'default_class': 'xmodule.hidden_module.HiddenDescriptor', 'direct': {
} 'ENGINE': 'xmodule.modulestore.mongo.MongoModuleStore',
'OPTIONS': modulestore_options
} }
} }
......
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