Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
E
edx-platform
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-platform
Commits
1fbb4b8b
Commit
1fbb4b8b
authored
Jun 21, 2016
by
Nimisha Asthagiri
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Update BlockStructure API
parent
f016647f
Hide whitespace changes
Inline
Side-by-side
Showing
8 changed files
with
255 additions
and
74 deletions
+255
-74
lms/djangoapps/course_api/blocks/serializers.py
+5
-1
lms/djangoapps/course_api/blocks/tests/test_serializers.py
+17
-3
lms/djangoapps/course_api/blocks/transformers/tests/test_block_counts.py
+5
-5
openedx/core/lib/block_structure/block_structure.py
+194
-55
openedx/core/lib/block_structure/cache.py
+2
-2
openedx/core/lib/block_structure/manager.py
+21
-6
openedx/core/lib/block_structure/tests/helpers.py
+7
-0
openedx/core/lib/block_structure/tests/test_block_structure.py
+4
-2
No files found.
lms/djangoapps/course_api/blocks/serializers.py
View file @
1fbb4b8b
...
...
@@ -17,10 +17,14 @@ class BlockSerializer(serializers.Serializer): # pylint: disable=abstract-metho
Get the field value requested. The field may be an XBlock field, a
transformer block field, or an entire tranformer block data dict.
"""
value
=
None
if
transformer
is
None
:
value
=
self
.
context
[
'block_structure'
]
.
get_xblock_field
(
block_key
,
field_name
)
elif
field_name
is
None
:
value
=
self
.
context
[
'block_structure'
]
.
get_transformer_block_data
(
block_key
,
transformer
)
try
:
value
=
self
.
context
[
'block_structure'
]
.
get_transformer_block_data
(
block_key
,
transformer
)
.
fields
except
KeyError
:
pass
else
:
value
=
self
.
context
[
'block_structure'
]
.
get_transformer_block_field
(
block_key
,
transformer
,
field_name
)
...
...
lms/djangoapps/course_api/blocks/tests/test_serializers.py
View file @
1fbb4b8b
...
...
@@ -41,10 +41,11 @@ class TestBlockSerializerBase(SharedModuleStoreTestCase):
block_types_to_count
=
[
'video'
],
requested_student_view_data
=
[
'video'
],
)
self
.
transformers
=
BlockStructureTransformers
(
COURSE_BLOCK_ACCESS_TRANSFORMERS
+
[
blocks_api_transformer
])
self
.
block_structure
=
get_course_blocks
(
self
.
user
,
self
.
course
.
location
,
BlockStructureTransformers
(
COURSE_BLOCK_ACCESS_TRANSFORMERS
+
[
blocks_api_transformer
])
,
self
.
transformers
,
)
self
.
serializer_context
=
{
'request'
:
MagicMock
(),
...
...
@@ -92,7 +93,7 @@ class TestBlockSerializerBase(SharedModuleStoreTestCase):
{
'id'
,
'type'
,
'lms_web_url'
,
'student_view_url'
,
'display_name'
,
'graded'
,
'
block_counts'
,
'
student_view_multi_device'
,
'student_view_multi_device'
,
'lti_url'
,
'visible_to_staff_only'
,
},
...
...
@@ -108,6 +109,13 @@ class TestBlockSerializerBase(SharedModuleStoreTestCase):
self
.
assertIn
(
'student_view_multi_device'
,
serialized_block
)
self
.
assertTrue
(
serialized_block
[
'student_view_multi_device'
])
# chapters with video should have block_counts
if
serialized_block
[
'type'
]
==
'chapter'
:
if
serialized_block
[
'display_name'
]
not
in
(
'poll_test'
,
'handout_container'
):
self
.
assertIn
(
'block_counts'
,
serialized_block
)
else
:
self
.
assertNotIn
(
'block_counts'
,
serialized_block
)
def
create_staff_context
(
self
):
"""
Create staff user and course blocks accessible by that user
...
...
@@ -119,7 +127,7 @@ class TestBlockSerializerBase(SharedModuleStoreTestCase):
block_structure
=
get_course_blocks
(
staff_user
,
self
.
course
.
location
,
BlockStructureTransformers
(
COURSE_BLOCK_ACCESS_TRANSFORMERS
)
,
self
.
transformers
,
)
return
{
'request'
:
MagicMock
(),
...
...
@@ -156,12 +164,14 @@ class TestBlockSerializer(TestBlockSerializerBase):
serializer
=
self
.
create_serializer
()
for
serialized_block
in
serializer
.
data
:
self
.
assert_basic_block
(
serialized_block
[
'id'
],
serialized_block
)
self
.
assertEquals
(
len
(
serializer
.
data
),
28
)
def
test_additional_requested_fields
(
self
):
self
.
add_additional_requested_fields
()
serializer
=
self
.
create_serializer
()
for
serialized_block
in
serializer
.
data
:
self
.
assert_extended_block
(
serialized_block
)
self
.
assertEquals
(
len
(
serializer
.
data
),
28
)
def
test_staff_fields
(
self
):
"""
...
...
@@ -173,6 +183,7 @@ class TestBlockSerializer(TestBlockSerializerBase):
for
serialized_block
in
serializer
.
data
:
self
.
assert_extended_block
(
serialized_block
)
self
.
assert_staff_fields
(
serialized_block
)
self
.
assertEquals
(
len
(
serializer
.
data
),
29
)
class
TestBlockDictSerializer
(
TestBlockSerializerBase
):
...
...
@@ -200,12 +211,14 @@ class TestBlockDictSerializer(TestBlockSerializerBase):
for
block_key_string
,
serialized_block
in
serializer
.
data
[
'blocks'
]
.
iteritems
():
self
.
assertEquals
(
serialized_block
[
'id'
],
block_key_string
)
self
.
assert_basic_block
(
block_key_string
,
serialized_block
)
self
.
assertEquals
(
len
(
serializer
.
data
[
'blocks'
]),
28
)
def
test_additional_requested_fields
(
self
):
self
.
add_additional_requested_fields
()
serializer
=
self
.
create_serializer
()
for
serialized_block
in
serializer
.
data
[
'blocks'
]
.
itervalues
():
self
.
assert_extended_block
(
serialized_block
)
self
.
assertEquals
(
len
(
serializer
.
data
[
'blocks'
]),
28
)
def
test_staff_fields
(
self
):
"""
...
...
@@ -217,3 +230,4 @@ class TestBlockDictSerializer(TestBlockSerializerBase):
for
serialized_block
in
serializer
.
data
[
'blocks'
]
.
itervalues
():
self
.
assert_extended_block
(
serialized_block
)
self
.
assert_staff_fields
(
serialized_block
)
self
.
assertEquals
(
len
(
serializer
.
data
[
'blocks'
]),
29
)
lms/djangoapps/course_api/blocks/transformers/tests/test_block_counts.py
View file @
1fbb4b8b
...
...
@@ -38,13 +38,13 @@ class TestBlockCountsTransformer(ModuleStoreTestCase):
)
# verify count of chapters
self
.
assertEquals
(
block_counts_for_course
[
'chapter'
]
,
2
)
self
.
assertEquals
(
block_counts_for_course
.
chapter
,
2
)
# verify count of problems
self
.
assertEquals
(
block_counts_for_course
[
'problem'
]
,
6
)
self
.
assertEquals
(
block_counts_for_chapter_x
[
'problem'
]
,
3
)
self
.
assertEquals
(
block_counts_for_course
.
problem
,
6
)
self
.
assertEquals
(
block_counts_for_chapter_x
.
problem
,
3
)
# verify other block types are not counted
for
block_type
in
[
'course'
,
'html'
,
'video'
]:
self
.
assert
NotIn
(
block_type
,
block_counts_for_course
)
self
.
assert
NotIn
(
block_type
,
block_counts_for_chapter_x
)
self
.
assert
False
(
hasattr
(
block_counts_for_course
,
block_type
)
)
self
.
assert
False
(
hasattr
(
block_counts_for_chapter_x
,
block_type
)
)
openedx/core/lib/block_structure/block_structure.py
View file @
1fbb4b8b
...
...
@@ -72,6 +72,9 @@ class BlockStructure(object):
"""
return
self
.
get_block_keys
()
def
__len__
(
self
):
return
len
(
self
.
_block_relations
)
#--- Block structure relation methods ---#
def
get_parents
(
self
,
usage_key
):
...
...
@@ -149,6 +152,7 @@ class BlockStructure(object):
self
,
filter_func
=
None
,
yield_descendants_of_unyielded
=
False
,
start_node
=
None
,
):
"""
Performs a topological sort of the block structure and yields
...
...
@@ -163,7 +167,7 @@ class BlockStructure(object):
traverse_topologically method.
"""
return
traverse_topologically
(
start_node
=
self
.
root_block_usage_key
,
start_node
=
s
tart_node
or
s
elf
.
root_block_usage_key
,
get_parents
=
self
.
get_parents
,
get_children
=
self
.
get_children
,
filter_func
=
filter_func
,
...
...
@@ -173,6 +177,7 @@ class BlockStructure(object):
def
post_order_traversal
(
self
,
filter_func
=
None
,
start_node
=
None
,
):
"""
Performs a post-order sort of the block structure and yields
...
...
@@ -187,7 +192,7 @@ class BlockStructure(object):
traverse_post_order method.
"""
return
traverse_post_order
(
start_node
=
self
.
root_block_usage_key
,
start_node
=
s
tart_node
or
s
elf
.
root_block_usage_key
,
get_children
=
self
.
get_children
,
filter_func
=
filter_func
,
)
...
...
@@ -267,19 +272,119 @@ class BlockStructure(object):
block_relations
[
usage_key
]
=
_BlockRelations
()
class
_Block
Data
(
object
):
class
Field
Data
(
object
):
"""
Data structure to encapsulate collected
data for a single block
.
Data structure to encapsulate collected
fields
.
"""
def
class_field_names
(
self
):
"""
Returns list of names of fields that are defined directly
on the class. Can be overridden by subclasses. All other
fields are assumed to be stored in the self.fields dict.
"""
return
[
'fields'
]
def
__init__
(
self
):
# Map of
xblock
field name to the field's value for this block.
# Map of field name to the field's value for this block.
# dict {string: any picklable type}
self
.
xblock_fields
=
{}
self
.
fields
=
{}
def
__getattr__
(
self
,
field_name
):
if
self
.
_is_own_field
(
field_name
):
return
super
(
FieldData
,
self
)
.
__getattr__
(
field_name
)
try
:
return
self
.
fields
[
field_name
]
except
KeyError
:
raise
AttributeError
(
"Field {0} does not exist"
.
format
(
field_name
))
def
__setattr__
(
self
,
field_name
,
field_value
):
if
self
.
_is_own_field
(
field_name
):
return
super
(
FieldData
,
self
)
.
__setattr__
(
field_name
,
field_value
)
else
:
self
.
fields
[
field_name
]
=
field_value
def
__delattr__
(
self
,
field_name
):
if
self
.
_is_own_field
(
field_name
):
return
super
(
FieldData
,
self
)
.
__delattr__
(
field_name
)
else
:
delattr
(
self
.
fields
,
field_name
)
def
_is_own_field
(
self
,
field_name
):
"""
Returns whether the given field_name is the name of an
actual field of this class.
"""
return
field_name
in
self
.
class_field_names
()
# Map of transformer name to the transformer's data for this
# block.
# defaultdict {string: dict}
self
.
transformer_data
=
defaultdict
(
dict
)
class
TransformerData
(
FieldData
):
"""
Data structure to encapsulate collected data for a transformer.
"""
pass
class
TransformerDataMap
(
dict
):
"""
A map of Transformer name to its corresponding TransformerData.
The map can be accessed by the Transformer's name or the
Transformer's class type.
"""
def
__getitem__
(
self
,
key
):
key
=
self
.
_translate_key
(
key
)
return
dict
.
__getitem__
(
self
,
key
)
def
__setitem__
(
self
,
key
,
value
):
key
=
self
.
_translate_key
(
key
)
dict
.
__setitem__
(
self
,
key
,
value
)
def
__delitem__
(
self
,
key
):
key
=
self
.
_translate_key
(
key
)
dict
.
__delitem__
(
self
,
key
)
def
get_or_create
(
self
,
key
):
"""
Returns the TransformerData associated with the given
key. If not found, creates and returns a new TransformerData
and maps it to the given key.
"""
try
:
return
self
[
key
]
except
KeyError
:
new_transformer_data
=
TransformerData
()
self
[
key
]
=
new_transformer_data
return
new_transformer_data
def
_translate_key
(
self
,
key
):
"""
Allows the given key to be either the transformer's class or name,
always returning the transformer's name. This allows
TransformerDataMap to be accessed in either of the following ways:
map[TransformerClass] or
map['transformer_name']
"""
try
:
return
key
.
name
()
except
AttributeError
:
return
key
class
BlockData
(
FieldData
):
"""
Data structure to encapsulate collected data for a single block.
"""
def
class_field_names
(
self
):
return
super
(
BlockData
,
self
)
.
class_field_names
()
+
[
'location'
,
'transformer_data'
]
def
__init__
(
self
,
usage_key
):
super
(
BlockData
,
self
)
.
__init__
()
# Location (or usage key) of the block.
self
.
location
=
usage_key
# Map of transformer name to its block-specific data.
self
.
transformer_data
=
TransformerDataMap
()
class
BlockStructureBlockData
(
BlockStructure
):
...
...
@@ -292,12 +397,31 @@ class BlockStructureBlockData(BlockStructure):
# Map of a block's usage key to its collected data, including
# its xBlock fields and block-specific transformer data.
# d
efaultdict {UsageKey: _
BlockData}
self
.
_block_data_map
=
defaultdict
(
_BlockData
)
# d
ict {UsageKey:
BlockData}
self
.
_block_data_map
=
{}
# Map of a transformer's name to its non-block-specific data.
# defaultdict {string: dict}
self
.
_transformer_data
=
defaultdict
(
dict
)
self
.
transformer_data
=
TransformerDataMap
()
def
iteritems
(
self
):
"""
Returns iterator of (UsageKey, BlockData) pairs for all
blocks in the BlockStructure.
"""
return
self
.
_block_data_map
.
iteritems
()
def
itervalues
(
self
):
"""
Returns iterator of BlockData for all blocks in the
BlockStructure.
"""
return
self
.
_block_data_map
.
itervalues
()
def
__getitem__
(
self
,
usage_key
):
"""
Returns the BlockData associated with the given key.
"""
return
self
.
_block_data_map
.
get
(
usage_key
)
def
get_xblock_field
(
self
,
usage_key
,
field_name
,
default
=
None
):
"""
...
...
@@ -316,7 +440,7 @@ class BlockStructureBlockData(BlockStructure):
not found.
"""
block_data
=
self
.
_block_data_map
.
get
(
usage_key
)
return
block_data
.
xblock_fields
.
get
(
field_name
,
default
)
if
block_data
else
default
return
getattr
(
block_data
,
field_name
,
default
)
if
block_data
else
default
def
get_transformer_data
(
self
,
transformer
,
key
,
default
=
None
):
"""
...
...
@@ -330,7 +454,10 @@ class BlockStructureBlockData(BlockStructure):
key (string) - A dictionary key to the transformer's data
that is requested.
"""
return
self
.
_transformer_data
.
get
(
transformer
.
name
(),
{})
.
get
(
key
,
default
)
try
:
return
getattr
(
self
.
transformer_data
[
transformer
],
key
,
default
)
except
KeyError
:
return
default
def
set_transformer_data
(
self
,
transformer
,
key
,
value
):
"""
...
...
@@ -346,7 +473,23 @@ class BlockStructureBlockData(BlockStructure):
value (any picklable type) - The value to associate with the
given key for the given transformer's data.
"""
self
.
_transformer_data
[
transformer
.
name
()][
key
]
=
value
setattr
(
self
.
transformer_data
.
get_or_create
(
transformer
),
key
,
value
)
def
get_transformer_block_data
(
self
,
usage_key
,
transformer
):
"""
Returns the TransformerData for the given
transformer for the block identified by the given usage_key.
Raises KeyError if not found.
Arguments:
usage_key (UsageKey) - Usage key of the block whose
transformer data is requested.
transformer (BlockStructureTransformer) - The transformer
whose dictionary data is requested.
"""
return
self
.
_block_data_map
[
usage_key
]
.
transformer_data
[
transformer
]
def
get_transformer_block_field
(
self
,
usage_key
,
transformer
,
key
,
default
=
None
):
"""
...
...
@@ -367,8 +510,11 @@ class BlockStructureBlockData(BlockStructure):
default (any type) - The value to return if a dictionary
entry is not found.
"""
transformer_data
=
self
.
get_transformer_block_data
(
usage_key
,
transformer
)
return
transformer_data
.
get
(
key
,
default
)
try
:
transformer_data
=
self
.
get_transformer_block_data
(
usage_key
,
transformer
)
except
KeyError
:
return
default
return
getattr
(
transformer_data
,
key
,
default
)
def
set_transformer_block_field
(
self
,
usage_key
,
transformer
,
key
,
value
):
"""
...
...
@@ -388,30 +534,11 @@ class BlockStructureBlockData(BlockStructure):
given key for the given transformer's data for the
requested block.
"""
self
.
_block_data_map
[
usage_key
]
.
transformer_data
[
transformer
.
name
()][
key
]
=
value
def
get_transformer_block_data
(
self
,
usage_key
,
transformer
):
"""
Returns the entire transformer data dict for the given
transformer for the block identified by the given usage_key;
returns an empty dict {} if not found.
Arguments:
usage_key (UsageKey) - Usage key of the block whose
transformer data is requested.
transformer (BlockStructureTransformer) - The transformer
whose dictionary data is requested.
key (string) - A dictionary key to the transformer's data
that is requested.
"""
default
=
{}
block_data
=
self
.
_block_data_map
.
get
(
usage_key
)
if
not
block_data
:
return
default
else
:
return
block_data
.
transformer_data
.
get
(
transformer
.
name
(),
default
)
setattr
(
self
.
_get_or_create_block
(
usage_key
)
.
transformer_data
.
get_or_create
(
transformer
),
key
,
value
,
)
def
remove_transformer_block_field
(
self
,
usage_key
,
transformer
,
key
):
"""
...
...
@@ -425,8 +552,11 @@ class BlockStructureBlockData(BlockStructure):
transformer (BlockStructureTransformer) - The transformer
whose data entry is to be deleted.
"""
transformer_block_data
=
self
.
get_transformer_block_data
(
usage_key
,
transformer
)
transformer_block_data
.
pop
(
key
,
None
)
try
:
transformer_block_data
=
self
.
get_transformer_block_data
(
usage_key
,
transformer
)
delattr
(
transformer_block_data
,
key
)
except
(
AttributeError
,
KeyError
):
pass
def
remove_block
(
self
,
usage_key
,
keep_descendants
):
"""
...
...
@@ -527,6 +657,19 @@ class BlockStructureBlockData(BlockStructure):
raise
TransformerException
(
'VERSION attribute is not set on transformer {0}.'
,
transformer
.
name
())
self
.
set_transformer_data
(
transformer
,
TRANSFORMER_VERSION_KEY
,
transformer
.
VERSION
)
def
_get_or_create_block
(
self
,
usage_key
):
"""
Returns the BlockData associated with the given usage_key.
If not found, creates and returns a new BlockData and
maps it to the given key.
"""
try
:
return
self
.
_block_data_map
[
usage_key
]
except
KeyError
:
block_data
=
BlockData
(
usage_key
)
self
.
_block_data_map
[
usage_key
]
=
block_data
return
block_data
class
BlockStructureModulestoreData
(
BlockStructureBlockData
):
"""
...
...
@@ -599,23 +742,19 @@ class BlockStructureModulestoreData(BlockStructureBlockData):
Iterates through all instantiated xBlocks that were added and
collects all xBlock fields that were requested.
"""
if
not
self
.
_requested_xblock_fields
:
return
for
xblock_usage_key
,
xblock
in
self
.
_xblock_map
.
iteritems
():
block_data
=
self
.
_get_or_create_block
(
xblock_usage_key
)
for
field_name
in
self
.
_requested_xblock_fields
:
self
.
_set_xblock_field
(
xblock_usage_key
,
xblock
,
field_name
)
self
.
_set_xblock_field
(
block_data
,
xblock
,
field_name
)
def
_set_xblock_field
(
self
,
usage_key
,
xblock
,
field_name
):
def
_set_xblock_field
(
self
,
block_data
,
xblock
,
field_name
):
"""
Updates the given block's xBlock fields data with the xBlock
value for the given field name.
Arguments:
usage_key (UsageKey) - Usage key of the given xBlock. This
value is passed in separately as opposed to retrieving
it from the given xBlock since this interface is
agnostic to and decoupled from the xBlock interface.
block_data (BlockData) - A BlockStructure BlockData
object.
xblock (XBlock) - An instantiated XBlock object whose
field is being accessed and collected for later
...
...
@@ -625,4 +764,4 @@ class BlockStructureModulestoreData(BlockStructureBlockData):
being collected and stored.
"""
if
hasattr
(
xblock
,
field_name
):
se
lf
.
_block_data_map
[
usage_key
]
.
xblock_fields
[
field_name
]
=
getattr
(
xblock
,
field_name
)
se
tattr
(
block_data
,
field_name
,
getattr
(
xblock
,
field_name
)
)
openedx/core/lib/block_structure/cache.py
View file @
1fbb4b8b
...
...
@@ -40,7 +40,7 @@ class BlockStructureCache(object):
"""
data_to_cache
=
(
block_structure
.
_block_relations
,
block_structure
.
_
transformer_data
,
block_structure
.
transformer_data
,
block_structure
.
_block_data_map
,
)
zp_data_to_cache
=
zpickle
(
data_to_cache
)
...
...
@@ -99,7 +99,7 @@ class BlockStructureCache(object):
block_relations
,
transformer_data
,
block_data_map
=
zunpickle
(
zp_data_from_cache
)
block_structure
=
BlockStructureModulestoreData
(
root_block_usage_key
)
block_structure
.
_block_relations
=
block_relations
block_structure
.
_
transformer_data
=
transformer_data
block_structure
.
transformer_data
=
transformer_data
block_structure
.
_block_data_map
=
block_data_map
return
block_structure
...
...
openedx/core/lib/block_structure/manager.py
View file @
1fbb4b8b
...
...
@@ -2,6 +2,8 @@
Top-level module for the Block Structure framework with a class for managing
BlockStructures.
"""
from
contextlib
import
contextmanager
from
.cache
import
BlockStructureCache
from
.factory
import
BlockStructureFactory
from
.exceptions
import
UsageKeyNotInBlockStructure
...
...
@@ -87,12 +89,13 @@ class BlockStructureManager(object):
)
cache_miss
=
block_structure
is
None
if
cache_miss
or
BlockStructureTransformers
.
is_collected_outdated
(
block_structure
):
block_structure
=
BlockStructureFactory
.
create_from_modulestore
(
self
.
root_block_usage_key
,
self
.
modulestore
)
BlockStructureTransformers
.
collect
(
block_structure
)
self
.
block_structure_cache
.
add
(
block_structure
)
with
self
.
_bulk_operations
():
block_structure
=
BlockStructureFactory
.
create_from_modulestore
(
self
.
root_block_usage_key
,
self
.
modulestore
)
BlockStructureTransformers
.
collect
(
block_structure
)
self
.
block_structure_cache
.
add
(
block_structure
)
return
block_structure
def
update_collected
(
self
):
...
...
@@ -111,3 +114,15 @@ class BlockStructureManager(object):
root block key.
"""
self
.
block_structure_cache
.
delete
(
self
.
root_block_usage_key
)
@contextmanager
def
_bulk_operations
(
self
):
"""
A context manager for notifying the store of bulk operations.
"""
try
:
course_key
=
self
.
root_block_usage_key
.
course_key
except
AttributeError
:
course_key
=
None
with
self
.
modulestore
.
bulk_operations
(
course_key
):
yield
openedx/core/lib/block_structure/tests/helpers.py
View file @
1fbb4b8b
...
...
@@ -68,6 +68,13 @@ class MockModulestore(object):
raise
ItemNotFoundError
return
item
@contextmanager
def
bulk_operations
(
self
,
ignore
):
# pylint: disable=unused-argument
"""
A context manager for notifying the store of bulk operations.
"""
yield
class
MockCache
(
object
):
"""
...
...
openedx/core/lib/block_structure/tests/test_block_structure.py
View file @
1fbb4b8b
...
...
@@ -138,17 +138,19 @@ class TestBlockStructureData(TestCase, ChildrenMapTestMixin):
# verify fields have not been collected yet
for
block
in
blocks
:
bs_block
=
block_structure
[
block
.
location
]
for
field
in
fields
:
self
.
assertIsNone
(
block_structure
.
get_xblock_field
(
block
.
location
,
field
))
self
.
assertIsNone
(
getattr
(
bs_block
,
field
,
None
))
# collect fields
block_structure
.
_collect_requested_xblock_fields
()
# verify values of collected fields
for
block
in
blocks
:
bs_block
=
block_structure
[
block
.
location
]
for
field
in
fields
:
self
.
assertEquals
(
block_structure
.
get_xblock_field
(
block
.
location
,
field
),
getattr
(
bs_block
,
field
,
None
),
block
.
field_map
.
get
(
field
),
)
...
...
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