Refactor error messages.

Error messages can come from different projects(cinder, keystone,
nova, etc.),but we are currently catching all the exceptions and
return a generic error message and it hard for the user to determine
the source of error. Horizon team decided to add a collapse-able box
for error messages which shows the detailed error message on the
horizon UI during shanghai summit[1]. This patch do the same.

Partially-Implements blueprint refactor-error-messages

[1] https://etherpad.openstack.org/p/horizon-u-ptg#110

Change-Id: If0bd24540562b8f1330ac6cb7db5f1d354e1d1b7
This commit is contained in:
manchandavishal 2020-02-17 08:46:45 +00:00
parent 484eee068e
commit 5081359295
7 changed files with 104 additions and 10 deletions

View File

@ -260,8 +260,13 @@ HANDLE_EXC_METHODS = [
]
def _append_detail(message, details):
return encoding.force_text(message) + '\u2026' + \
encoding.force_text(details)
def handle(request, message=None, redirect=None, ignore=False,
escalate=False, log_level=None, force_log=None):
escalate=False, log_level=None, force_log=None, details=None):
"""Centralized error handling for Horizon.
Because Horizon consumes so many different APIs with completely
@ -288,6 +293,9 @@ def handle(request, message=None, redirect=None, ignore=False,
If the exception is not re-raised, an appropriate wrapper exception
class indicating the type of exception that was encountered will be
returned.
If details is None (default), take it from exception sys.exc_info.
If details is other string, then use that string explicitly or if details
is empty then suppress it.
"""
exc_type, exc_value, exc_traceback = sys.exc_info()
log_method = getattr(LOG, log_level or "exception")
@ -316,6 +324,10 @@ def handle(request, message=None, redirect=None, ignore=False,
user_message = encoding.force_text(message) % {"exc": log_entry}
elif message:
user_message = encoding.force_text(message)
if details is None:
user_message = _append_detail(user_message, exc_value)
elif details:
user_message = _append_detail(user_message, details)
for exc_handler in HANDLE_EXC_METHODS:
if issubclass(exc_type, exc_handler['exc']):

View File

@ -12,8 +12,11 @@
* under the License.
*/
horizon.alert = function (type, message, extra_tags) {
horizon.alert = function (type, message, extra_tags, details) {
var safe = false;
var arr = extractDetail(message);
var message = arr[0];
var details = arr[1];
// Check if the message is tagged as safe.
if (typeof(extra_tags) !== "undefined" && $.inArray('safe', extra_tags.split(' ')) !== -1) {
safe = true;
@ -32,12 +35,17 @@ horizon.alert = function (type, message, extra_tags) {
type = 'danger';
}
function extractDetail(str) {
return str.split('\u2026');
}
var template = horizon.templates.compiled_templates["#alert_message_template"],
params = {
"type": type || 'default',
"type_display": type_display,
"message": message,
"safe": safe
"safe": safe,
"details": details
};
var this_alert = $(template.render(params)).hide().prependTo("#main_content .messages").fadeIn(100);
horizon.autoDismissAlert(this_alert);

View File

@ -1,4 +1,4 @@
{% load i18n %}
{% load i18n splitfilter %}
<div class="messages">
<toast></toast>
{% for message in messages %}
@ -31,8 +31,19 @@
<a class="close" data-dismiss="alert" href="#">
<span class="fa fa-times"></span>
</a>
<p><strong>{% trans "Error: " %}</strong>{{ message }}</p>
<p>
{% with message.message|split_message as messages %}
<strong>{% trans "Error: " %}</strong>{{ messages|first }}
{% if messages|length > 1 %}
<a href="#message_details" data-toggle="collapse"
data-target="#message_details">Details</a>
<div id="message_details" class="collapse">
{{ messages|last }}
</div>
{% endif %}
</p>
</div>
{% endwith %}
{% endif %}
{% endfor %}
</div>
</div>

View File

@ -15,6 +15,10 @@
[[/safe]]
[[^safe]]
[[message]]
<a href="#message_details" data-toggle="collapse" data-target="#message_details">Details</a>
<div id="message_details" class="collapse">
[[details]]
</div>
[[/safe]]
</p>
</div>

View File

@ -0,0 +1,20 @@
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
from django import template
register = template.Library()
@register.filter(name='split_message')
def split_message(value):
return value.split('\u2026')

View File

@ -25,7 +25,8 @@ class HandleTests(test.TestCase):
# Japanese translation of:
# 'Because the container is not empty, it can not be deleted.'
expected = ['error', force_text(translated_unicode), '']
expected = ['error', force_text(translated_unicode +
'\u2026' + translated_unicode), '']
req = self.request
req.META['HTTP_X_REQUESTED_WITH'] = 'XMLHttpRequest'
@ -57,6 +58,35 @@ class HandleTests(test.TestCase):
# message part of the first message. There should be only one message
# in this test case.
self.assertIn(message, req.horizon['async_messages'][0][1])
# verifies that the exec message which in this case is not trusted
# is not in the message content.
self.assertNotIn(exc_msg, req.horizon['async_messages'][0][1])
# verifies that the exec message which in this case is in the
# message content.
self.assertIn(exc_msg, req.horizon['async_messages'][0][1])
def test_handle_exception_with_empty_details(self):
message = u"Couldn't make the thing"
details = ""
expected = ['error', message, '']
req = self.request
req.META['HTTP_X_REQUESTED_WITH'] = 'XMLHttpRequest'
try:
raise Exception(message)
except Exception:
exceptions.handle(req, message, details=details)
self.assertCountEqual(req.horizon['async_messages'], [expected])
def test_handle_exception_with_details(self):
message = u"Couldn't make the thing"
exc_msg = u"Exception string"
details = "custom detail message"
expected = ['error', message + '\u2026' + details, '']
req = self.request
req.META['HTTP_X_REQUESTED_WITH'] = 'XMLHttpRequest'
try:
raise Exception(exc_msg)
except Exception:
exceptions.handle(req, message, details=details)
self.assertCountEqual(req.horizon['async_messages'], [expected])

View File

@ -0,0 +1,9 @@
---
features:
- |
[`blueprint refactor-error-messages <https://blueprints.launchpad.net/horizon/+spec/refactor-error-messages>`_]
User can see detailed error message on the horizon UI.
This blueprint adds a hyperlink "details" in the alert box.
So now when an exception occurs a user can click on this hyperlink
"details" which shows original error message included in a corresponding
error. This may help a user to understand what happens in detail.