From 8a1929d916cad7f1aa0cc8e933d5700b6e6230b2 Mon Sep 17 00:00:00 2001 From: Andrejs Verza Date: Thu, 13 Aug 2020 13:48:06 +0300 Subject: fixed performace and memory usage in Latest data view --- ChangeLog.d/bugfix/ZBX-18071 | 1 + ui/app/controllers/CControllerLatest.php | 214 ++++++++---------- ui/app/controllers/CControllerLatestView.php | 2 + .../CControllerLatestViewRefresh.php | 2 + .../classes/api/managers/CHistoryManager.php | 4 +- 5 files changed, 107 insertions(+), 116 deletions(-) create mode 100644 ChangeLog.d/bugfix/ZBX-18071 diff --git a/ChangeLog.d/bugfix/ZBX-18071 b/ChangeLog.d/bugfix/ZBX-18071 new file mode 100644 index 0000000000..7cd7263145 --- /dev/null +++ b/ChangeLog.d/bugfix/ZBX-18071 @@ -0,0 +1 @@ +A......... [ZBX-18071] fixed slow SQL query when filtering items with latest data available (averza) diff --git a/ui/app/controllers/CControllerLatest.php b/ui/app/controllers/CControllerLatest.php index 419b608faa..ac3a458c52 100644 --- a/ui/app/controllers/CControllerLatest.php +++ b/ui/app/controllers/CControllerLatest.php @@ -25,16 +25,18 @@ abstract class CControllerLatest extends CController { /** - * Prepares the latest data based on the given filter and sorting options. + * Prepare the latest data based on the given filter and sorting options. * - * @param array $filter Item filter options. - * @param array $filter['groupids'] Filter items by host groups. - * @param array $filter['hostids'] Filter items by hosts. - * @param string $filter['application'] Filter items by application. - * @param string $filter['select'] Filter items by name. - * @param int $filter['show_without_data'] Include items with empty history. - * @param string $sort_field Sorting field. - * @param string $sort_order Sorting order. + * @param array $filter Item filter options. + * @param array $filter['groupids'] Filter items by host groups. + * @param array $filter['hostids'] Filter items by hosts. + * @param string $filter['application'] Filter items by application. + * @param string $filter['select'] Filter items by name. + * @param int $filter['show_without_data'] Include items with empty history. + * @param string $sort_field Sorting field. + * @param string $sort_order Sorting order. + * + * @return array */ protected function prepareData(array $filter, $sort_field, $sort_order) { $config = select_config(); @@ -95,121 +97,88 @@ abstract class CControllerLatest extends CController { // Select hosts for subsequent selection of applications and items. - if ($filter['hostids']) { - $hosts = API::Host()->get([ - 'output' => ['hostid', 'name', 'status'], - 'groupids' => $groupids, - 'hostids' => $filter['hostids'], - 'with_monitored_items' => true, - 'preservekeys' => true - ]); - - $hostids = array_keys($hosts); - } - else { - $hosts = null; - $hostids = null; - } - - // Select applications for subsequent selection of items. - - if ($filter['application'] !== '') { - $applications = API::Application()->get([ - 'output' => ['applicationid', 'name'], - 'groupids' => $groupids, - 'hostids' => $hostids, - 'templated' => false, - 'search' => ['name' => $filter['application']], - 'preservekeys' => true - ]); - - $applicationids = array_keys($applications); - } - else { - $applications = null; - $applicationids = null; - } - - // Select unlimited items based on filter, requesting minimum data. - $items = API::Item()->get([ - 'output' => ['itemid', 'hostid', 'value_type'], + $hosts = API::Host()->get([ + 'output' => ['hostid', 'name', 'status'], 'groupids' => $groupids, - 'hostids' => $hostids, - 'applicationids' => $applicationids, - 'webitems' => true, - 'templated' => false, - 'filter' => [ - 'status' => [ITEM_STATUS_ACTIVE] - ], - 'search' => ($filter['select'] === '') ? null : [ - 'name' => $filter['select'] - ], + 'hostids' => $filter['hostids'] ? $filter['hostids'] : null, + 'monitored_hosts' => true, 'preservekeys' => true ]); - if ($items) { - if ($hosts === null) { - $hosts = API::Host()->get([ - 'output' => ['hostid', 'name', 'status'], - 'groupids' => $groupids, - 'hostids' => array_keys(array_flip(array_column($items, 'hostid'))), - 'with_monitored_items' => true, + CArrayHelper::sort($hosts, [$host_sort_options]); + $hostids = array_keys($hosts); + $hostids_index = array_flip($hostids); + + $applications = []; + + $select_hosts = []; + $select_items = []; + + foreach ($hosts as $hostid => $host) { + if ($filter['application'] !== '') { + $host_applications = API::Application()->get([ + 'output' => ['applicationid', 'name'], + 'hostids' => [$hostid], + 'search' => ['name' => $filter['application']], 'preservekeys' => true ]); - } - - CArrayHelper::sort($hosts, [$host_sort_options]); - $hostids = array_keys($hosts); - $items_of_hosts = []; + $host_applicationids = array_keys($host_applications); - foreach ($items as $itemid => $item) { - $items_of_hosts[$item['hostid']][$itemid] = $item; + $applications += $host_applications; + } + else { + $host_applicationids = null; } - uksort($items_of_hosts, function($hostid_1, $hostid_2) use ($hostids) { - return (array_search($hostid_1, $hostids, true) <=> array_search($hostid_2, $hostids, true)); - }); + $host_items = API::Item()->get([ + 'output' => ['itemid', 'hostid', 'value_type'], + 'hostids' => [$hostid], + 'applicationids' => $host_applicationids, + 'webitems' => true, + 'filter' => [ + 'status' => [ITEM_STATUS_ACTIVE] + ], + 'search' => ($filter['select'] === '') ? null : [ + 'name' => $filter['select'] + ], + 'preservekeys' => true + ]); - $select_items = []; + $select_hosts[$hostid] = true; - foreach ($items_of_hosts as $host_items) { - $select_items += $filter['show_without_data'] - ? $host_items - : Manager::History()->getItemsHavingValues($host_items, ZBX_HISTORY_PERIOD); + $select_items += $filter['show_without_data'] + ? $host_items + : Manager::History()->getItemsHavingValues($host_items, ZBX_HISTORY_PERIOD); - if (count($select_items) > $config['search_limit']) { - break; - } + if (count($select_items) > $config['search_limit']) { + break; } + } - // Select limited set of items, requesting extended data. + if ($select_items) { + // Select items, requesting extended data. $items = API::Item()->get([ 'output' => ['itemid', 'type', 'hostid', 'name', 'key_', 'delay', 'history', 'trends', 'status', 'value_type', 'units', 'valuemapid', 'description', 'state', 'error' ], 'selectApplications' => ['applicationid'], - 'groupids' => $groupids, - 'hostids' => $hostids, - 'applicationids' => $applicationids, 'itemids' => array_keys($select_items), 'webitems' => true, 'preservekeys' => true ]); - if ($applications === null) { + if ($filter['application'] === '') { $applications = API::Application()->get([ 'output' => ['applicationid', 'name'], - 'groupids' => $groupids, - 'hostids' => $hostids, - 'itemids' => array_keys($items), - 'templated' => false, + 'hostids' => array_keys($select_hosts), 'preservekeys' => true ]); } CArrayHelper::sort($applications, [$application_sort_options]); $applicationids = array_keys($applications); + $applicationids_index = array_flip($applicationids); $applications_size = []; $items_grouped = []; @@ -239,21 +208,23 @@ abstract class CControllerLatest extends CController { } } - $items = []; $rows = []; + $items = []; - uksort($items_grouped, function($hostid_1, $hostid_2) use ($hostids) { - return (array_search($hostid_1, $hostids, true) <=> array_search($hostid_2, $hostids, true)); + uksort($items_grouped, function($hostid_1, $hostid_2) use ($hostids_index) { + return ($hostids_index[$hostid_1] <=> $hostids_index[$hostid_2]); }); foreach ($items_grouped as $host_items_grouped) { - uksort($host_items_grouped, function($id_1, $id_2) use ($applicationids, $application_sort_options) { - if ($id_1 == 0 || $id_2 == 0) { - return bccomp($id_1, $id_2) * (($application_sort_options['order'] === 'ASC') ? -1 : 1); - } + uksort($host_items_grouped, + function($id_1, $id_2) use ($applicationids_index, $application_sort_options) { + if ($id_1 == 0 || $id_2 == 0) { + return bccomp($id_1, $id_2) * (($application_sort_options['order'] === 'ASC') ? -1 : 1); + } - return (array_search($id_1, $applicationids, true) <=> array_search($id_2, $applicationids, true)); - }); + return ($applicationids_index[$id_1] <=> $applicationids_index[$id_2]); + } + ); foreach ($host_items_grouped as $applicationid => $application_items) { CArrayHelper::sort($application_items, [$item_sort_options]); @@ -261,36 +232,26 @@ abstract class CControllerLatest extends CController { foreach ($application_items as $itemid => $item) { unset($item['applications']); - $items[$itemid] = $item; $rows[] = [ 'itemid' => $itemid, 'applicationid' => $applicationid ]; + $items[$itemid] = $item; + if (count($rows) > $config['search_limit']) { break 3; } } } } - - // Resolve macros. - - $items = CMacrosResolverHelper::resolveItemKeys($items); - $items = CMacrosResolverHelper::resolveItemNames($items); - $items = CMacrosResolverHelper::resolveTimeUnitMacros($items, ['delay', 'history', 'trends']); - - // Choosing max history period for already filtered items having data. - $history_period = $filter['show_without_data'] ? ZBX_HISTORY_PERIOD : null; - - $history = Manager::History()->getLastValues($items, 2, $history_period); } else { $rows = []; $hosts = []; $applications = []; $applications_size = []; - $history = []; + $items = []; } if ($filter['hostids']) { @@ -313,9 +274,34 @@ abstract class CControllerLatest extends CController { 'applications' => $applications, 'applications_size' => $applications_size, 'items' => $items, - 'history' => $history, 'multiselect_hostgroup_data' => $multiselect_hostgroup_data, 'multiselect_host_data' => $multiselect_host_data ]; } + + /** + * Extend previously prepared data. + * + * @param array $prepared_data Data returned by prepareData method. + * @param int $show_without_data Include items with empty history. + */ + protected function extendData(array &$prepared_data, $show_without_data) { + $items = array_intersect_key($prepared_data['items'], + array_flip(array_column($prepared_data['rows'], 'itemid')) + ); + + // Resolve macros. + + $items = CMacrosResolverHelper::resolveItemKeys($items); + $items = CMacrosResolverHelper::resolveItemNames($items); + $items = CMacrosResolverHelper::resolveTimeUnitMacros($items, ['delay', 'history', 'trends']); + + // Choosing max history period for already filtered items having data. + $history_period = $show_without_data ? ZBX_HISTORY_PERIOD : null; + + $history = Manager::History()->getLastValues($items, 2, $history_period); + + $prepared_data['items'] = $items; + $prepared_data['history'] = $history; + } } diff --git a/ui/app/controllers/CControllerLatestView.php b/ui/app/controllers/CControllerLatestView.php index f6465d76e6..1fc9351a36 100644 --- a/ui/app/controllers/CControllerLatestView.php +++ b/ui/app/controllers/CControllerLatestView.php @@ -121,6 +121,8 @@ class CControllerLatestView extends CControllerLatest { $paging = CPagerHelper::paginate(getRequest('page', 1), $prepared_data['rows'], ZBX_SORT_UP, $view_curl); + $this->extendData($prepared_data, $filter['show_without_data']); + // display $data = [ 'filter' => $filter, diff --git a/ui/app/controllers/CControllerLatestViewRefresh.php b/ui/app/controllers/CControllerLatestViewRefresh.php index 6da872d6d8..eea2fa73ab 100644 --- a/ui/app/controllers/CControllerLatestViewRefresh.php +++ b/ui/app/controllers/CControllerLatestViewRefresh.php @@ -79,6 +79,8 @@ class CControllerLatestViewRefresh extends CControllerLatest { $paging = CPagerHelper::paginate(getRequest('page', 1), $prepared_data['rows'], ZBX_SORT_UP, $view_curl); + $this->extendData($prepared_data, $filter['show_without_data']); + // display $data = [ 'filter' => $filter, diff --git a/ui/include/classes/api/managers/CHistoryManager.php b/ui/include/classes/api/managers/CHistoryManager.php index 8cbfb141d0..a16a6f3725 100644 --- a/ui/include/classes/api/managers/CHistoryManager.php +++ b/ui/include/classes/api/managers/CHistoryManager.php @@ -74,8 +74,8 @@ class CHistoryManager { 'SELECT itemid'. ' FROM '.self::getTableName($type). ' WHERE '.dbConditionInt('itemid', $type_itemids). - ' GROUP BY itemid'. - ($period ? ' HAVING MAX(clock)>'.$period : '') + ($period ? ' AND clock>'.$period : ''). + ' GROUP BY itemid' ), 'itemid'); $results += array_intersect_key($items, array_flip($type_results)); -- 2.21.0.windows.1