createTabs(); $this->view->compact = $this->_request->getParam('view') === 'compact'; if ($this->_request->getParam('view') === 'inline') { $this->view->compact = true; $this->view->inline = true; } } /** * @deprecated DO NOT USE. THIS IS A HACK. This is removed once we fix the eventhistory action w/ filters. */ protected function applyFilter($query) { $params = clone $this->params; $params->shift('format'); $params->shift('limit'); $params->shift('page'); $params->shift('view'); if ($sort = $params->shift('sort')) { $query->order($sort, $params->shift('dir')); } $query->applyFilter(Filter::fromQuerystring((string) $params)); $this->handleFormatRequest($query); return $query; } protected function hasBetterUrl() { $request = $this->getRequest(); $url = Url::fromRequest(); if ($this->getRequest()->isPost()) { if ($request->getPost('sort')) { $url->setParam('sort', $request->getPost('sort')); if ($request->getPost('dir')) { $url->setParam('dir', $request->getPost('dir')); } else { $url->removeParam('dir'); } return $url; } } return false; } /** * Overwrite the backend to use (used for testing) * * @param Backend $backend The Backend that should be used for querying */ public function setBackend($backend) { $this->backend = $backend; } /** * Display host list */ public function hostsAction() { if ($url = $this->hasBetterUrl()) { return $this->redirectNow($url); } // Handle soft and hard states if (strtolower($this->params->shift('stateType', 'soft')) === 'hard') { $stateColumn = 'host_hard_state'; $stateChangeColumn = 'host_last_hard_state_change'; } else { $stateColumn = 'host_state'; $stateChangeColumn = 'host_last_state_change'; } $this->addTitleTab('hosts', $this->translate('Hosts'), $this->translate('List hosts')); $this->setAutorefreshInterval(10); $query = $this->backend->select()->from('hostStatus', array_merge(array( 'host_icon_image', 'host_name', 'host_display_name', 'host_state' => $stateColumn, 'host_address', 'host_acknowledged', 'host_output', 'host_attempt', 'host_in_downtime', 'host_is_flapping', 'host_state_type', 'host_handled', 'host_last_check', 'host_last_state_change' => $stateChangeColumn, 'host_notifications_enabled', 'host_unhandled_services', 'host_action_url', 'host_notes_url', 'host_last_comment', 'host_last_ack', 'host_last_downtime', 'host_active_checks_enabled', 'host_passive_checks_enabled', 'host_current_check_attempt', 'host_max_check_attempts' ), $this->extraColumns())); $this->filterQuery($query); $this->applyRestriction('monitoring/hosts/filter', $query); $this->setupSortControl(array( 'host_severity' => $this->translate('Severity'), 'host_state' => $this->translate('Current State'), 'host_display_name' => $this->translate('Hostname'), 'host_address' => $this->translate('Address'), 'host_last_check' => $this->translate('Last Check') )); $this->view->hosts = $query->paginate(); $this->view->stats = $this->backend->select()->from('statusSummary', array( 'hosts_total', 'hosts_up', 'hosts_down', 'hosts_down_handled', 'hosts_down_unhandled', 'hosts_unreachable', 'hosts_unreachable_handled', 'hosts_unreachable_unhandled', 'hosts_pending', ))->getQuery()->fetchRow(); } /** * Display service list */ public function servicesAction() { if ($url = $this->hasBetterUrl()) { return $this->redirectNow($url); } // Handle soft and hard states if (strtolower($this->params->shift('stateType', 'soft')) === 'hard') { $stateColumn = 'service_hard_state'; $stateChangeColumn = 'service_last_hard_state_change'; } else { $stateColumn = 'service_state'; $stateChangeColumn = 'service_last_state_change'; } $this->addTitleTab('services', $this->translate('Services'), $this->translate('List services')); $this->view->showHost = true; if ($host = $this->_getParam('host')) { if (strpos($host, '*') === false) { $this->view->showHost = false; } } $this->setAutorefreshInterval(10); $columns = array_merge(array( 'host_name', 'host_display_name', 'host_state', 'host_state_type', 'host_last_state_change', 'host_address', 'host_handled', 'service_description', 'service_display_name', 'service_state' => $stateColumn, 'service_in_downtime', 'service_acknowledged', 'service_handled', 'service_output', 'service_perfdata', 'service_attempt', 'service_last_state_change' => $stateChangeColumn, 'service_icon_image', 'service_is_flapping', 'service_state_type', 'service_handled', 'service_severity', 'service_last_check', 'service_notifications_enabled', 'service_action_url', 'service_notes_url', 'service_last_comment', 'service_last_ack', 'service_last_downtime', 'service_active_checks_enabled', 'service_passive_checks_enabled', 'current_check_attempt' => 'service_current_check_attempt', 'max_check_attempts' => 'service_max_check_attempts' ), $this->extraColumns()); $query = $this->backend->select()->from('serviceStatus', $columns); $this->filterQuery($query); $this->applyRestriction('monitoring/services/filter', $query); $this->setupSortControl(array( 'service_severity' => $this->translate('Service Severity'), 'service_state' => $this->translate('Current Service State'), 'service_display_name' => $this->translate('Service Name'), 'service_last_check' => $this->translate('Last Service Check'), 'host_severity' => $this->translate('Host Severity'), 'host_state' => $this->translate('Current Host State'), 'host_display_name' => $this->translate('Hostname'), 'host_address' => $this->translate('Host Address'), 'host_last_check' => $this->translate('Last Host Check') )); $limit = $this->params->get('limit'); $this->view->limit = $limit; if ($limit === 0) { $this->view->services = $query->getQuery()->fetchAll(); } else { // TODO: Workaround, paginate should be able to fetch limit from new params $this->view->services = $query->paginate($this->params->get('limit')); } $this->view->stats = $this->backend->select()->from('statusSummary', array( 'services_total', 'services_ok', 'services_problem', 'services_problem_handled', 'services_problem_unhandled', 'services_critical', 'services_critical_unhandled', 'services_critical_handled', 'services_warning', 'services_warning_unhandled', 'services_warning_handled', 'services_unknown', 'services_unknown_unhandled', 'services_unknown_handled', 'services_pending', ))->getQuery()->fetchRow(); } /** * Fetch the current downtimes and put them into the view property `downtimes` */ public function downtimesAction() { if ($url = $this->hasBetterUrl()) { return $this->redirectNow($url); } $this->addTitleTab('downtimes', $this->translate('Downtimes'), $this->translate('List downtimes')); $this->setAutorefreshInterval(12); $query = $this->backend->select()->from('downtime', array( 'id' => 'downtime_internal_id', 'objecttype' => 'downtime_objecttype', 'comment' => 'downtime_comment', 'author' => 'downtime_author', 'start' => 'downtime_start', 'scheduled_start' => 'downtime_scheduled_start', 'scheduled_end' => 'downtime_scheduled_end', 'end' => 'downtime_end', 'duration' => 'downtime_duration', 'is_flexible' => 'downtime_is_flexible', 'is_fixed' => 'downtime_is_fixed', 'is_in_effect' => 'downtime_is_in_effect', 'entry_time' => 'downtime_entry_time', 'host' => 'downtime_host', 'service' => 'downtime_service', 'host_state' => 'downtime_host_state', 'service_state' => 'downtime_service_state', 'host_display_name', 'service_display_name' )); $this->filterQuery($query); $this->setupSortControl(array( 'downtime_is_in_effect' => $this->translate('Is In Effect'), 'host_display_name' => $this->translate('Host'), 'service_display_name' => $this->translate('Service'), 'downtime_entry_time' => $this->translate('Entry Time'), 'downtime_author' => $this->translate('Author'), 'downtime_start' => $this->translate('Start Time'), 'downtime_end' => $this->translate('End Time'), 'downtime_scheduled_start' => $this->translate('Scheduled Start'), 'downtime_scheduled_end' => $this->translate('Scheduled End'), 'downtime_duration' => $this->translate('Duration') )); $this->view->downtimes = $query->paginate(); if ($this->Auth()->hasPermission('monitoring/command/downtime/delete')) { $this->view->delDowntimeForm = new DeleteDowntimeCommandForm(); } } /** * Display notification overview */ public function notificationsAction() { if ($url = $this->hasBetterUrl()) { return $this->redirectNow($url); } $this->addTitleTab( 'notifications', $this->translate('Notifications'), $this->translate('List notifications') ); $this->setAutorefreshInterval(15); $query = $this->backend->select()->from('notification', array( 'host', 'service', 'notification_output', 'notification_contact', 'notification_start_time', 'notification_state', 'host_display_name', 'service_display_name' )); $this->filterQuery($query); $this->view->notifications = $query->paginate(); $this->setupSortControl(array( 'notification_start_time' => $this->translate('Notification Start') )); } public function contactsAction() { if ($url = $this->hasBetterUrl()) { return $this->redirectNow($url); } $this->addTitleTab('contacts', $this->translate('Contacts'), $this->translate('List contacts')); $query = $this->backend->select()->from('contact', array( 'contact_name', 'contact_id', 'contact_alias', 'contact_email', 'contact_pager', 'contact_notify_service_timeperiod', 'contact_notify_service_recovery', 'contact_notify_service_warning', 'contact_notify_service_critical', 'contact_notify_service_unknown', 'contact_notify_service_flapping', 'contact_notify_service_downtime', 'contact_notify_host_timeperiod', 'contact_notify_host_recovery', 'contact_notify_host_down', 'contact_notify_host_unreachable', 'contact_notify_host_flapping', 'contact_notify_host_downtime', )); $this->filterQuery($query); $this->view->contacts = $query->paginate(); $this->setupSortControl(array( 'contact_name' => $this->translate('Name'), 'contact_alias' => $this->translate('Alias'), 'contact_email' => $this->translate('Email'), 'contact_pager' => $this->translate('Pager Address / Number'), 'contact_notify_service_timeperiod' => $this->translate('Service Notification Timeperiod'), 'contact_notify_host_timeperiod' => $this->translate('Host Notification Timeperiod') )); } public function eventgridAction() { if ($url = $this->hasBetterUrl()) { return $this->redirectNow($url); } $this->addTitleTab('eventgrid', $this->translate('Event Grid'), $this->translate('Show the Event Grid')); $form = new StatehistoryForm(); $form->setEnctype(Zend_Form::ENCTYPE_URLENCODED); $form->setMethod('get'); $form->setTokenDisabled(); $form->setUidDisabled(); $form->render(); $this->view->form = $form; $this->params->remove('view'); $orientation = $this->params->shift('vertical', 0) ? 'vertical' : 'horizontal'; /* $orientationBox = new SelectBox( 'orientation', array( '0' => mt('monitoring', 'Vertical'), '1' => mt('monitoring', 'Horizontal') ), mt('monitoring', 'Orientation'), 'horizontal' ); $orientationBox->applyRequest($this->getRequest()); */ $query = $this->backend->select()->from( 'eventgrid', array('day', $form->getValue('state')) ); $this->params->remove(array('objecttype', 'from', 'to', 'state', 'btn_submit')); $this->view->filter = Filter::fromQuerystring((string) $this->params); $query->applyFilter($this->view->filter); $this->view->summary = $query->getQuery()->fetchAll(); $this->view->column = $form->getValue('state'); // $this->view->orientationBox = $orientationBox; $this->view->orientation = $orientation; } public function contactgroupsAction() { if ($url = $this->hasBetterUrl()) { return $this->redirectNow($url); } $this->addTitleTab( 'contactgroups', $this->translate('Contact Groups'), $this->translate('List contact groups') ); $query = $this->backend->select()->from('contactgroup', array( 'contactgroup_name', 'contactgroup_alias', 'contact_name', 'contact_alias', 'contact_email', 'contact_pager', ))->order('contactgroup_alias'); $this->filterQuery($query); // Fetch and prepare all contact groups: $contactgroups = $query->getQuery()->fetchAll(); $groupData = array(); foreach ($contactgroups as $c) { if (!array_key_exists($c->contactgroup_name, $groupData)) { $groupData[$c->contactgroup_name] = array( 'alias' => $c->contactgroup_alias, 'contacts' => array() ); } $groupData[$c->contactgroup_name]['contacts'][] = $c; } // TODO: Find a better naming $this->view->groupData = $groupData; } public function commentsAction() { if ($url = $this->hasBetterUrl()) { return $this->redirectNow($url); } $this->addTitleTab('comments', $this->translate('Comments'), $this->translate('List comments')); $this->setAutorefreshInterval(12); $query = $this->backend->select()->from('comment', array( 'id' => 'comment_internal_id', 'objecttype' => 'comment_objecttype', 'comment' => 'comment_data', 'author' => 'comment_author', 'timestamp' => 'comment_timestamp', 'type' => 'comment_type', 'persistent' => 'comment_is_persistent', 'expiration' => 'comment_expiration', 'host' => 'comment_host', 'service' => 'comment_service', 'host_display_name', 'service_display_name' )); $this->filterQuery($query); $this->view->comments = $query->paginate(); $this->setupSortControl( array( 'comment_timestamp' => $this->translate('Comment Timestamp'), 'host_display_name' => $this->translate('Host'), 'service_display_name' => $this->translate('Service'), 'comment_type' => $this->translate('Comment Type'), 'comment_expiration' => $this->translate('Expiration') ) ); if ($this->Auth()->hasPermission('monitoring/command/comment/delete')) { $this->view->delCommentForm = new DeleteCommentCommandForm(); } } public function servicegroupsAction() { if ($url = $this->hasBetterUrl()) { return $this->redirectNow($url); } $this->addTitleTab( 'servicegroups', $this->translate('Service Groups'), $this->translate('List service groups') ); $this->setAutorefreshInterval(12); $query = $this->backend->select()->from('groupsummary', array( 'servicegroup', 'servicegroup_alias', 'hosts_up', 'hosts_unreachable_handled', 'hosts_unreachable_unhandled', 'hosts_down_handled', 'hosts_down_unhandled', 'hosts_pending', 'services_ok', 'services_unknown_handled', 'services_unknown_unhandled', 'services_critical_handled', 'services_critical_unhandled', 'services_warning_handled', 'services_warning_unhandled', 'services_pending', 'services_ok_last_state_change', 'services_pending_last_state_change', 'services_warning_last_state_change_handled', 'services_critical_last_state_change_handled', 'services_unknown_last_state_change_handled', 'services_warning_last_state_change_unhandled', 'services_critical_last_state_change_unhandled', 'services_unknown_last_state_change_unhandled', 'services_total' ))->order('services_severity')->order('servicegroup_alias'); // TODO(el): Can't default to the sort rules of the data view because it's meant for both host groups and // service groups. We should separate them. $this->filterQuery($query); $this->view->servicegroups = $query->paginate(); $this->setupSortControl(array( 'services_severity' => $this->translate('Severity'), 'servicegroup_alias' => $this->translate('Service Group Name'), 'services_total' => $this->translate('Total Services'), 'services_ok' => $this->translate('Services OK'), 'services_unknown' => $this->translate('Services UNKNOWN'), 'services_critical' => $this->translate('Services CRITICAL'), 'services_warning' => $this->translate('Services WARNING'), 'services_pending' => $this->translate('Services PENDING') )); } public function hostgroupsAction() { if ($url = $this->hasBetterUrl()) { return $this->redirectNow($url); } $this->addTitleTab('hostgroups', $this->translate('Host Groups'), $this->translate('List host groups')); $this->setAutorefreshInterval(12); $query = $this->backend->select()->from('groupsummary', array( 'hostgroup', 'hostgroup_alias', 'hosts_up', 'hosts_unreachable_handled', 'hosts_unreachable_unhandled', 'hosts_down_handled', 'hosts_down_unhandled', 'hosts_pending', 'services_ok', 'services_unknown_handled', 'services_unknown_unhandled', 'services_critical_handled', 'services_critical_unhandled', 'services_warning_handled', 'services_warning_unhandled', 'services_pending', 'services_ok_last_state_change', 'services_pending_last_state_change', 'services_warning_last_state_change_handled', 'services_critical_last_state_change_handled', 'services_unknown_last_state_change_handled', 'services_warning_last_state_change_unhandled', 'services_critical_last_state_change_unhandled', 'services_unknown_last_state_change_unhandled', 'services_total' ))->order('services_severity')->order('hostgroup_alias'); // TODO(el): Can't default to the sort rules of the data view because it's meant for both host groups and // service groups. We should separate them. $this->filterQuery($query); $this->view->hostgroups = $query->paginate(); $this->setupSortControl(array( 'services_severity' => $this->translate('Severity'), 'hostgroup_alias' => $this->translate('Host Group Name'), 'services_total' => $this->translate('Total Services'), 'services_ok' => $this->translate('Services OK'), 'services_unknown' => $this->translate('Services UNKNOWN'), 'services_critical' => $this->translate('Services CRITICAL'), 'services_warning' => $this->translate('Services WARNING'), 'services_pending' => $this->translate('Services PENDING') )); } public function eventhistoryAction() { if ($url = $this->hasBetterUrl()) { return $this->redirectNow($url); } $this->addTitleTab( 'eventhistory', $this->translate('Event Overview'), $this->translate('List event records') ); $query = $this->backend->select()->from('eventHistory', array( 'host_name', 'host_display_name', 'service_description', 'service_display_name', 'object_type', 'timestamp', 'state', 'attempt', 'max_attempts', 'output', 'type', 'host', 'service' )); $this->filterQuery($query); $this->setupSortControl(array( 'timestamp' => $this->translate('Occurence') )); $this->view->history = $query->paginate(); } public function servicegridAction() { if ($url = $this->hasBetterUrl()) { return $this->redirectNow($url); } $this->addTitleTab('servicegrid', $this->translate('Service Grid'), $this->translate('Show the Service Grid')); $this->setAutorefreshInterval(15); $query = $this->backend->select()->from('serviceStatus', array( 'host_name', 'service_description', 'service_state', 'service_output', 'service_handled' )); $this->filterQuery($query); $this->setupSortControl(array( 'host_name' => $this->translate('Hostname'), 'service_description' => $this->translate('Service description') )); $pivot = $query->pivot('service_description', 'host_name'); $this->view->pivot = $pivot; $this->view->horizontalPaginator = $pivot->paginateXAxis(); $this->view->verticalPaginator = $pivot->paginateYAxis(); } protected function filterQuery($query) { $editor = Widget::create('filterEditor') ->setQuery($query) ->preserveParams('limit', 'sort', 'dir', 'format', 'view', 'backend', 'stateType', 'addColumns') ->ignoreParams('page') ->handleRequest($this->getRequest()); $query->applyFilter($editor->getFilter()); $this->view->filterEditor = $editor; $this->view->filter = $editor->getFilter(); if ($sort = $this->params->get('sort')) { $query->order($sort, $this->params->get('dir')); } $this->handleFormatRequest($query); return $query; } protected function extraColumns() { $columns = preg_split( '~,~', $this->params->shift('addcolumns', ''), -1, PREG_SPLIT_NO_EMPTY ); $this->view->extraColumns = $columns; return $columns; } /** * Create a sort control box at the 'sortControl' view parameter * * @param array $columns An array containing the sort columns, with the * submit value as the key and the value as the label */ private function setupSortControl(array $columns) { $this->view->sortControl = new SortBox( 'sortbox-' . $this->getRequest()->getActionName(), $columns ); $this->view->sortControl->applyRequest($this->getRequest()); } protected function addTitleTab($action, $title, $tip) { $this->getTabs()->add($action, array( 'title' => $tip, 'label' => $title, 'url' => Url::fromRequest() ))->activate($action); $this->view->title = $title; } /** * Return all tabs for this controller * * @return Tabs */ private function createTabs() { $tabs = $this->getTabs(); $tabs->setTitle($this->translate('Monitoring Navigation')); if (in_array($this->_request->getActionName(), array( 'hosts', 'services', 'eventhistory', 'eventgrid', 'notifications' ))) { $tabs->extend(new OutputFormat())->extend(new DashboardAction()); } } }