Commit dcd74e6d by Calen Pennington

Make abtests work, using the new abtest xml format

parent c3a432f2
...@@ -2,11 +2,10 @@ import json ...@@ -2,11 +2,10 @@ import json
import random import random
from lxml import etree from lxml import etree
from x_module import XModule, XModuleDescriptor from xmodule.x_module import XModule
from xmodule.raw_module import RawDescriptor
from xmodule.xml_module import XmlDescriptor
class ModuleDescriptor(XModuleDescriptor): from xmodule.exceptions import InvalidDefinitionError
pass
def group_from_value(groups, v): def group_from_value(groups, v):
...@@ -25,7 +24,7 @@ def group_from_value(groups, v): ...@@ -25,7 +24,7 @@ def group_from_value(groups, v):
return g return g
class Module(XModule): class ABTestModule(XModule):
""" """
Implements an A/B test with an aribtrary number of competing groups Implements an A/B test with an aribtrary number of competing groups
...@@ -37,20 +36,14 @@ class Module(XModule): ...@@ -37,20 +36,14 @@ class Module(XModule):
</abtest> </abtest>
""" """
def __init__(self, system, xml, item_id, instance_state=None, shared_state=None): def __init__(self, system, location, definition, instance_state=None, shared_state=None, **kwargs):
XModule.__init__(self, system, xml, item_id, instance_state, shared_state) XModule.__init__(self, system, location, definition, instance_state, shared_state, **kwargs)
self.xmltree = etree.fromstring(xml)
target_groups = self.xmltree.findall('group') target_groups = self.definition['data'].keys()
if shared_state is None: if shared_state is None:
target_values = [
(elem.get('name'), float(elem.get('portion')))
for elem in target_groups
]
default_value = 1 - sum(val for (_, val) in target_values)
self.group = group_from_value( self.group = group_from_value(
target_values + [(None, default_value)], self.definition['data']['group_portions'],
random.uniform(0, 1) random.uniform(0, 1)
) )
else: else:
...@@ -69,24 +62,57 @@ class Module(XModule): ...@@ -69,24 +62,57 @@ class Module(XModule):
self.group = shared_state['group'] self.group = shared_state['group']
def get_shared_state(self): def get_shared_state(self):
print self.group
return json.dumps({'group': self.group}) return json.dumps({'group': self.group})
def _xml_children(self): def displayable_items(self):
group = None return [self.system.get_module(child)
if self.group is None: for child
group = self.xmltree.find('default') in self.definition['data']['group_content'][self.group]]
else:
for candidate_group in self.xmltree.find('group'):
if self.group == candidate_group.get('name'): class ABTestDescriptor(RawDescriptor, XmlDescriptor):
group = candidate_group module_class = ABTestModule
break
def __init__(self, system, definition=None, **kwargs):
kwargs['shared_state_key'] = definition['data']['experiment']
RawDescriptor.__init__(self, system, definition, **kwargs)
@classmethod
def definition_from_xml(cls, xml_object, system):
experiment = xml_object.get('experiment')
if experiment is None:
raise InvalidDefinitionError("ABTests must specify an experiment. Not found in:\n{xml}".format(xml=etree.tostring(xml_object, pretty_print=True)))
definition = {
'data': {
'experiment': experiment,
'group_portions': [],
'group_content': {None: []},
},
'children': []}
for group in xml_object:
if group.tag == 'default':
name = None
else:
name = group.get('name')
definition['data']['group_portions'].append(
(name, float(group.get('portion', 0)))
)
child_content_urls = [
system.process_xml(etree.tostring(child)).url
for child in group
]
definition['data']['group_content'][name] = child_content_urls
definition['children'].extend(child_content_urls)
if group is None: default_portion = 1 - sum(portion for (name, portion) in definition['data']['group_portions'])
return [] if default_portion < 0:
return list(group) raise InvalidDefinitionError("ABTest portions must add up to less than or equal to 1")
def get_children(self): definition['data']['group_portions'].append((None, default_portion))
return [self.module_from_xml(child) for child in self._xml_children()]
def get_html(self): return definition
return '\n'.join(child.get_html() for child in self.get_children())
class InvalidDefinitionError(Exception):
pass
...@@ -13,6 +13,7 @@ setup( ...@@ -13,6 +13,7 @@ setup(
# for a description of entry_points # for a description of entry_points
entry_points={ entry_points={
'xmodule.v1': [ 'xmodule.v1': [
"abtest = xmodule.abtest_module:ABTestDescriptor",
"book = xmodule.translation_module:TranslateCustomTagDescriptor", "book = xmodule.translation_module:TranslateCustomTagDescriptor",
"chapter = xmodule.seq_module:SequenceDescriptor", "chapter = xmodule.seq_module:SequenceDescriptor",
"course = xmodule.seq_module:SequenceDescriptor", "course = xmodule.seq_module:SequenceDescriptor",
......
...@@ -295,6 +295,7 @@ class XModuleDescriptor(Plugin): ...@@ -295,6 +295,7 @@ class XModuleDescriptor(Plugin):
self.display_name = kwargs.get('display_name') self.display_name = kwargs.get('display_name')
self.format = kwargs.get('format') self.format = kwargs.get('format')
self.graded = kwargs.get('graded', False) self.graded = kwargs.get('graded', False)
self.shared_state_key = kwargs.get('shared_state_key')
# For now, we represent goals as a list of strings, but this # For now, we represent goals as a list of strings, but this
# is one of the things that we are going to be iterating on heavily # is one of the things that we are going to be iterating on heavily
......
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