diff --git a/searx/metrics/__init__.py b/searx/metrics/__init__.py
index 9648e6215..5665ca63c 100644
--- a/searx/metrics/__init__.py
+++ b/searx/metrics/__init__.py
@@ -97,12 +97,12 @@ def initialize(engine_names=None):
histogram_storage.configure(histogram_width, histogram_size, 'engine', engine_name, 'time', 'total')
-def get_engine_errors(engline_list):
+def get_engine_errors(engline_name_list):
result = {}
engine_names = list(errors_per_engines.keys())
engine_names.sort()
for engine_name in engine_names:
- if engine_name not in engline_list:
+ if engine_name not in engline_name_list:
continue
error_stats = errors_per_engines[engine_name]
@@ -126,61 +126,86 @@ def get_engine_errors(engline_list):
return result
-def to_percentage(stats, maxvalue):
- for engine_stat in stats:
- if maxvalue:
- engine_stat['percentage'] = int(engine_stat['avg'] / maxvalue * 100)
+def get_reliabilities(engline_name_list, checker_results):
+ reliabilities = {}
+
+ engine_errors = get_engine_errors(engline_name_list)
+
+ for engine_name in engline_name_list:
+ checker_result = checker_results.get(engine_name, {})
+ checker_success = checker_result.get('success', True)
+ errors = engine_errors.get(engine_name) or []
+ if counter('engine', engine_name, 'search', 'count', 'sent') == 0:
+ # no request
+ reliablity = None
+ elif checker_success and not errors:
+ reliablity = 100
+ elif 'simple' in checker_result.get('errors', {}):
+ # the basic (simple) test doesn't work: the engine is broken accoding to the checker
+ # even if there is no exception
+ reliablity = 0
else:
- engine_stat['percentage'] = 0
- return stats
+ reliablity = 100 - sum([error['percentage'] for error in errors if not error.get('secondary')])
+
+ reliabilities[engine_name] = {
+ 'reliablity': reliablity,
+ 'errors': errors,
+ 'checker': checker_results.get(engine_name, {}).get('errors', {}).keys(),
+ }
+ return reliabilities
-def get_engines_stats(engine_list):
+def round_or_none(number, digits):
+ return round(number, digits) if number else number
+
+
+def get_engines_stats(engine_name_list):
assert counter_storage is not None
assert histogram_storage is not None
list_time = []
max_time_total = max_result_count = None # noqa
- for engine_name in engine_list:
- successful_count = counter('engine', engine_name, 'search', 'count', 'successful')
- if successful_count == 0:
+ for engine_name in engine_name_list:
+ sent_count = counter('engine', engine_name, 'search', 'count', 'sent')
+ if sent_count == 0:
continue
- result_count_sum = histogram('engine', engine_name, 'result', 'count').sum
+ successful_count = counter('engine', engine_name, 'search', 'count', 'successful')
+
time_total = histogram('engine', engine_name, 'time', 'total').percentage(50)
time_http = histogram('engine', engine_name, 'time', 'http').percentage(50)
time_total_p80 = histogram('engine', engine_name, 'time', 'total').percentage(80)
time_http_p80 = histogram('engine', engine_name, 'time', 'http').percentage(80)
time_total_p95 = histogram('engine', engine_name, 'time', 'total').percentage(95)
time_http_p95 = histogram('engine', engine_name, 'time', 'http').percentage(95)
- result_count = result_count_sum / float(successful_count)
- if result_count:
+ result_count = histogram('engine', engine_name, 'result', 'count').percentage(50)
+ result_count_sum = histogram('engine', engine_name, 'result', 'count').sum
+ if successful_count and result_count_sum:
score = counter('engine', engine_name, 'score') # noqa
score_per_result = score / float(result_count_sum)
else:
score = score_per_result = 0.0
- max_time_total = max(time_total, max_time_total or 0)
- max_result_count = max(result_count, max_result_count or 0)
+ max_time_total = max(time_total or 0, max_time_total or 0)
+ max_result_count = max(result_count or 0, max_result_count or 0)
list_time.append({
- 'total': round(time_total, 1),
- 'total_p80': round(time_total_p80, 1),
- 'total_p95': round(time_total_p95, 1),
- 'http': round(time_http, 1),
- 'http_p80': round(time_http_p80, 1),
- 'http_p95': round(time_http_p95, 1),
'name': engine_name,
- 'processing': round(time_total - time_http, 1),
- 'processing_p80': round(time_total_p80 - time_http_p80, 1),
- 'processing_p95': round(time_total_p95 - time_http_p95, 1),
+ 'total': round_or_none(time_total, 1),
+ 'total_p80': round_or_none(time_total_p80, 1),
+ 'total_p95': round_or_none(time_total_p95, 1),
+ 'http': round_or_none(time_http, 1),
+ 'http_p80': round_or_none(time_http_p80, 1),
+ 'http_p95': round_or_none(time_http_p95, 1),
+ 'processing': round(time_total - time_http, 1) if time_total else None,
+ 'processing_p80': round(time_total_p80 - time_http_p80, 1) if time_total else None,
+ 'processing_p95': round(time_total_p95 - time_http_p95, 1) if time_total else None,
'score': score,
'score_per_result': score_per_result,
'result_count': result_count,
})
-
return {
'time': list_time,
'max_time': math.ceil(max_time_total or 0),
diff --git a/searx/static/themes/oscar/css/pointhi.css b/searx/static/themes/oscar/css/pointhi.css
index 64f612d79..99ed7b576 100644
--- a/searx/static/themes/oscar/css/pointhi.css
+++ b/searx/static/themes/oscar/css/pointhi.css
@@ -682,6 +682,7 @@ input[type=checkbox]:not(:checked) + .label_hide_if_checked + .label_hide_if_not
padding: 0.5rem 1rem;
margin: 0rem 0 0 2rem;
border: 1px solid #ddd;
+ box-shadow: 2px 2px 2px 0px rgba(0, 0, 0, 0.1);
background: white;
font-size: 14px;
font-weight: normal;
@@ -756,3 +757,21 @@ td:hover .engine-tooltip,
padding: 0.4rem 0;
width: 1px;
}
+.stacked-bar-chart-serie1 {
+ display: flex;
+ flex-shrink: 0;
+ flex-grow: 0;
+ flex-basis: unset;
+ background: #5bc0de;
+ box-shadow: inset 0 -1px 0 rgba(0, 0, 0, 0.15);
+ padding: 0.4rem 0;
+}
+.stacked-bar-chart-serie2 {
+ display: flex;
+ flex-shrink: 0;
+ flex-grow: 0;
+ flex-basis: unset;
+ background: #deb15b;
+ box-shadow: inset 0 -1px 0 rgba(0, 0, 0, 0.15);
+ padding: 0.4rem 0;
+}
diff --git a/searx/static/themes/oscar/css/pointhi.min.css b/searx/static/themes/oscar/css/pointhi.min.css
index 4332e4767..80027ff8e 100644
Binary files a/searx/static/themes/oscar/css/pointhi.min.css and b/searx/static/themes/oscar/css/pointhi.min.css differ
diff --git a/searx/static/themes/oscar/css/pointhi.min.css.map b/searx/static/themes/oscar/css/pointhi.min.css.map
index abb30817f..708e2f63a 100644
Binary files a/searx/static/themes/oscar/css/pointhi.min.css.map and b/searx/static/themes/oscar/css/pointhi.min.css.map differ
diff --git a/searx/static/themes/oscar/src/less/pointhi/preferences.less b/searx/static/themes/oscar/src/less/pointhi/preferences.less
index cb63674ed..352aed513 100644
--- a/searx/static/themes/oscar/src/less/pointhi/preferences.less
+++ b/searx/static/themes/oscar/src/less/pointhi/preferences.less
@@ -8,6 +8,7 @@
padding: 0.5rem 1rem;
margin: 0rem 0 0 2rem;
border: 1px solid #ddd;
+ box-shadow: 2px 2px 2px 0px rgba(0,0,0,0.1);
background: white;
font-size: 14px;
font-weight: normal;
@@ -77,3 +78,17 @@ th:hover .engine-tooltip, td:hover .engine-tooltip, .engine-tooltip:hover {
width: 1px;
}
+.stacked-bar-chart-serie1 {
+ .stacked-bar-chart-base();
+ background: #5bc0de;
+ box-shadow: inset 0 -1px 0 rgba(0,0,0,.15);
+ padding: 0.4rem 0;
+}
+
+.stacked-bar-chart-serie2 {
+ .stacked-bar-chart-base();
+ background: #deb15b;
+ box-shadow: inset 0 -1px 0 rgba(0,0,0,.15);
+ padding: 0.4rem 0;
+}
+
diff --git a/searx/templates/oscar/stats.html b/searx/templates/oscar/stats.html
index 0851343ce..149522226 100644
--- a/searx/templates/oscar/stats.html
+++ b/searx/templates/oscar/stats.html
@@ -1,6 +1,5 @@
{% extends "oscar/base.html" %}
{% block styles %}
-
{% endblock %}
{% block title %}{{ _('stats') }} - {% endblock %}
+
+{%- macro th_sort(column_order, column_name) -%}
+ {% if column_order==sort_order %}
+ {{ column_name }} {{ icon('chevron-down') }}
+ {% else %}
+ {{ column_name }}
+ {% endif %}
+{%- endmacro -%}
+
{% block content %}
{{ _('Scores per result') }}: {{ engine_stat.score_per_result | round(3) }}{{ _('Engine stats') }}
@@ -25,27 +33,33 @@
{% else %}
-
{% for engine_stat in engine_stats.get('time', []) %}
{{ _("Engine name") }}
- {{ _('Scores') }}
- {{ _('Number of results') }}
- {{ _('Response time') }}
+ {{ th_sort('name', _("Engine name")) }}
+ {{ th_sort('score', _('Scores')) }}
+ {{ th_sort('result_count', _('Result count')) }}
+ {{ th_sort('time', _('Response time')) }}
+ {{ th_sort('reliability', _('Reliability')) }}
{{ engine_stat.name }}
+ {% if engine_stat.score %}
{{ engine_stat.score|round(1) }}
+ {%- if engine_stat.result_count -%}
{{- "" -}}
{{- "" -}}
+
+ {%- endif -%}
+ {%- if engine_stat.total -%}
{{- "" -}}