Skip to content

Commit

Permalink
feat: [AXIMST-490] display render errors (#2506)
Browse files Browse the repository at this point in the history
* feat: [AXIMST-490] display render errors

* fix: resolve discussions
  • Loading branch information
ruzniaievdm committed Apr 4, 2024
1 parent a128bf1 commit 2b56a4c
Show file tree
Hide file tree
Showing 4 changed files with 68 additions and 32 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ class MessageValidation(serializers.Serializer):
Serializer for representing XBlock error.
"""

text = serializers.CharField()
text = serializers.CharField()
type = serializers.CharField()


Expand Down Expand Up @@ -114,8 +114,8 @@ class ChildVerticalContainerSerializer(serializers.Serializer):
user_partition_info = serializers.DictField()
user_partitions = serializers.ListField()
actions = serializers.SerializerMethodField()
has_validation_error = serializers.BooleanField()
validation_errors = MessageValidation(many=True)
validation_messages = MessageValidation(many=True)
render_error = serializers.CharField()

def get_actions(self, obj): # pylint: disable=unused-argument
"""
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -198,8 +198,8 @@ def test_children_content(self):
"actions": {
"can_manage_tags": True,
},
"has_validation_error": False,
"validation_errors": [],
"validation_messages": [],
"render_error": "",
},
{
"name": self.html_unit_second.display_name_with_default,
Expand All @@ -210,8 +210,8 @@ def test_children_content(self):
"actions": {
"can_manage_tags": True,
},
"has_validation_error": False,
"validation_errors": [],
"validation_messages": [],
"render_error": "",
},
]
self.assertEqual(response.data["children"], expected_response)
Expand Down Expand Up @@ -270,13 +270,8 @@ def test_validation_errors(self):
response = self.client.get(url)
children_response = response.data["children"]

# Check for an error in html_unit_first xblock
self.assertTrue(children_response[0]["has_validation_error"])
# Verify that html_unit_first access settings contradict its parent's access settings.
self.assertEqual(children_response[0]["validation_messages"][0]["type"], ValidationMessage.ERROR)

# Verify that html access settings contradict its parent's access settings.
validation = html_unit_first.validate()
self.assertEqual(len(validation.messages), 1)
self.assertEqual(validation.to_json()['messages'][0]['type'], ValidationMessage.ERROR)

# Check for an error in html_unit_second xblock
self.assertFalse(children_response[1]["has_validation_error"])
# Verify that html_unit_second has no validation messages.
self.assertFalse(children_response[1]["validation_messages"])
20 changes: 11 additions & 9 deletions cms/djangoapps/contentstore/rest_api/v1/views/vertical_block.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,8 @@
get_container_handler_context,
get_user_partition_info,
get_visibility_partition_info,
get_validation_messages,
get_xblock_validation_messages,
get_xblock_render_error,
)
from cms.djangoapps.contentstore.views.component import _get_item_in_course
from cms.djangoapps.contentstore.xblock_storage_handlers.view_handlers import get_xblock
Expand Down Expand Up @@ -206,8 +207,8 @@ def get(self, request: Request, usage_key_string: str):
"actions": {
"can_manage_tags": true,
}
"has_validation_error": false,
"validation_errors": [],
"validation_messages": [],
"render_error": "",
},
{
"name": "Text",
Expand All @@ -218,13 +219,13 @@ def get(self, request: Request, usage_key_string: str):
"actions": {
"can_manage_tags": true,
},
"has_validation_error": true,
"validation_errors": [
"validation_messages": [
{
"text": "This component's access settings contradict its parent's access settings.",
"type": "error"
}
]
],
"render_error": "Unterminated control keyword: 'if' in file '../problem.html'",
},
],
"is_published": false,
Expand All @@ -243,16 +244,17 @@ def get(self, request: Request, usage_key_string: str):
child_info = modulestore().get_item(child)
user_partition_info = get_visibility_partition_info(child_info, course=course)
user_partitions = get_user_partition_info(child_info, course=course)
validation_errors, has_validation_error = get_validation_messages(child_info)
validation_messages = get_xblock_validation_messages(child_info)
render_error = get_xblock_render_error(request, child_info)

children.append({
"name": child_info.display_name_with_default,
"block_id": child_info.location,
"block_type": child_info.location.block_type,
"user_partition_info": user_partition_info,
"user_partitions": user_partitions,
"has_validation_error": has_validation_error,
"validation_errors": validation_errors,
"validation_messages": validation_messages,
"render_error": render_error,
})

is_published = not modulestore().has_changes(current_xblock)
Expand Down
53 changes: 46 additions & 7 deletions cms/djangoapps/contentstore/utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -2236,19 +2236,58 @@ def send_course_update_notification(course_key, content, user):
COURSE_NOTIFICATION_REQUESTED.send_event(course_notification_data=notification_data)


def get_validation_messages(xblock):
def get_xblock_validation_messages(xblock):
"""
Retrieves validation messages for a given xblock.
Args:
xblock: The xblock object to validate.
Returns:
tuple:
- validation_errors (list): A list of validation error messages.
- has_validation_error (bool): True if there are validation errors, False otherwise.
list: A list of validation error messages.
"""
validation_json = xblock.validate().to_json()
validation_errors = validation_json['messages']
has_validation_error = bool(validation_errors)
return validation_errors, has_validation_error
return validation_json['messages']


def get_xblock_render_error(request, xblock):
"""
Checks if there are any rendering errors for a given block and return these.
Args:
request: WSGI request object
xblock: The xblock object to rendering.
Returns:
str: Error message which happened while rendering of xblock.
"""
from cms.djangoapps.contentstore.views.preview import _load_preview_block
from xmodule.studio_editable import has_author_view
from xmodule.x_module import AUTHOR_VIEW, STUDENT_VIEW

def get_xblock_render_context(request, block):
"""
Return a dict of the data needs for render of each block.
"""
can_edit = has_studio_write_access(request.user, block.usage_key.course_key)

return {
"is_unit_page": False,
"can_edit": can_edit,
"root_xblock": xblock,
"reorderable_items": set(),
"paging": None,
"force_render": None,
"item_url": "/container/{block.location}",
"tags_count_map": {},
}

try:
block = _load_preview_block(request, xblock)
preview_view = AUTHOR_VIEW if has_author_view(block) else STUDENT_VIEW
render_context = get_xblock_render_context(request, block)
block.render(preview_view, render_context)
except Exception as exc: # pylint: disable=broad-except
return str(exc)

return ""

0 comments on commit 2b56a4c

Please sign in to comment.