setBreadcrum([]); $this->task = []; $this->msg = $msg; $this->icon = $icon; $this->label = __($label); $this->page = $page; $this->url = ui_get_full_url( 'index.php?sec=gservers&sec2=godmode/servers/discovery&wiz=tasklist' ); return $this; } /** * Implements run method. * * @param string $message Redirected input. * @param boolean $status Redirected input. * * @return mixed Returns null if wizard is ongoing. Result if done. */ public function run($message='', $status=null) { global $config; // Load styles. parent::run(); $this->prepareBreadcrum( [ [ 'link' => 'index.php?sec=gservers&sec2=godmode/servers/discovery', 'label' => 'Discovery', ], ] ); // Header. ui_print_page_header( __('Task list'), '', false, '', true, '', false, '', GENERIC_SIZE_TEXT, '', $this->printHeader(true) ); // Show redirected messages from discovery.php. if ($status === 0) { ui_print_success_message($message); } else if ($status !== null) { ui_print_error_message($message); } $force_run = (bool) get_parameter('force_run'); $force = (bool) get_parameter('force'); if ($force_run === true || $force === true) { return $this->forceTask(); } $delete_console_task = (bool) get_parameter('delete_console_task'); if ($delete_console_task === true) { return $this->deleteConsoleTask(); } $toggle_console_task = (int) get_parameter('toggle_console_task', -1); if ($toggle_console_task === 1 || $toggle_console_task === 0) { return $this->toggleConsoleTask($toggle_console_task); } $delete = (bool) get_parameter('delete', false); if ($delete === true) { return $this->deleteTask(); } $disable = (bool) get_parameter('disabled', false); if ($disable === true) { return $this->disableTask(); } $enable = (bool) get_parameter('enabled', false); if ($enable === true) { return $this->enableTask(); } if (enterprise_installed()) { // This check only applies to enterprise users. enterprise_hook('tasklist_checkrunning'); $ret = $this->showListConsoleTask(); } else { $ret = false; } if (is_reporting_console_node() === false) { $ret2 = $this->showList(__('Host & devices tasks'), [0, 1]); if (enterprise_installed()) { $ret2 .= $this->showList(__('Applications tasks'), [3, 4, 5, 10, 11, 12], 'app'); $ret2 .= $this->showList(__('Cloud tasks'), [6, 7, 8, 13, 14], 'cloud'); $ret2 .= $this->showList(__('Custom tasks'), [-1], 'custom'); } } if ($ret === false && $ret2 === false) { include_once $config['homedir'].'/general/first_task/recon_view.php'; } else { $form = [ 'form' => [ 'method' => 'POST', 'action' => ui_get_full_url( 'index.php?sec=gservers&sec2=godmode/servers/discovery' ), 'class' => 'flex_center', ], 'inputs' => [ [ 'arguments' => [ 'name' => 'submit', 'label' => __('Go back'), 'type' => 'submit', 'attributes' => [ 'icon' => 'back', 'mode' => 'secondary', ], 'return' => true, ], ], [ 'arguments' => [ 'name' => 'refresh', 'label' => __('Refresh'), 'type' => 'button', 'attributes' => [ 'icon' => 'cog'], 'return' => true, 'script' => 'location.href = "'.$this->url.'";', ], ], ], ]; html_print_action_buttons($this->printForm($form, true)); } return $ret; } /** * Implements load method. * * @return mixed Skeleton for button. */ public function load() { return [ 'icon' => $this->icon, 'label' => $this->label, 'url' => $this->url, ]; } /** * Delete a recon task. * * @return void */ public function deleteTask() { global $config; if (! check_acl($config['id_user'], 0, 'AW')) { db_pandora_audit( AUDIT_LOG_ACL_VIOLATION, 'Trying to access recon task viewer' ); include 'general/noaccess.php'; return; } $task = get_parameter('task', null); if ($task !== null) { $result = db_process_sql_delete( 'trecon_task', ['id_rt' => $task] ); if ($result == 1) { return [ 'result' => 0, 'msg' => __('Task successfully deleted'), 'id' => false, ]; } // Trick to avoid double execution. header('Location: '.$this->url); } } /** * Force console task. * * @return void */ public function forceTask() { global $config; if (!$this->aclMulticheck('RR|RW|RM|PM')) { db_pandora_audit( AUDIT_LOG_ACL_VIOLATION, 'Trying to access recon task viewer' ); include 'general/noaccess.php'; return; } $id_console_task = (int) get_parameter('id_console_task'); if ($id_console_task != null) { // -------------------------------- // FORCE A CONSOLE TASK // -------------------------------- enterprise_hook('cron_task_run', [$id_console_task, true]); // Trick to avoid double execution. header('Location: '.$this->url); } else { // -------------------------------- // FORCE A RECON TASK // -------------------------------- if (check_acl($config['id_user'], 0, 'AW')) { if (isset($_GET['force'])) { $id = (int) get_parameter_get('force', 0); // Schedule execution. $review_mode = db_get_value( 'review_mode', 'trecon_task', 'id_rt', $id ); if ($review_mode != DISCOVERY_STANDARD) { // Force re-scan for supervised tasks. $review_mode = DISCOVERY_REVIEW; } db_process_sql_update( 'trecon_task', [ 'utimestamp' => 0, 'status' => 1, 'review_mode' => $review_mode, ], ['id_rt' => $id] ); header('Location: '.$this->url); } } } } /** * Toggle enable/disable status of selected Console Task. * * @param integer $enable If 1 enable the console task. * * @return void */ public function toggleConsoleTask(int $enable) { global $config; if ((bool) check_acl($config['id_user'], 0, 'RM') === false) { db_pandora_audit( AUDIT_LOG_ACL_VIOLATION, 'Trying to access recon task viewer' ); include 'general/noaccess.php'; return; } $id_console_task = (int) get_parameter('id_console_task'); if ($id_console_task > 0) { $result = db_process_sql_update( 'tuser_task_scheduled', ['enabled' => $enable], ['id' => $id_console_task] ); if ((int) $result === 1) { return [ 'result' => 0, 'msg' => ((bool) $enable === true) ? __('Task successfully enabled') : __('Task succesfully disabled'), 'id' => false, ]; } // Trick to avoid double execution. header('Location: '.$this->url); } } /** * Delete a Console task. * * @return void */ public function deleteConsoleTask() { global $config; if (! check_acl($config['id_user'], 0, 'RM')) { db_pandora_audit( AUDIT_LOG_ACL_VIOLATION, 'Trying to access recon task viewer' ); include 'general/noaccess.php'; return; } $id_console_task = (int) get_parameter('id_console_task'); if ($id_console_task !== null) { $result = db_process_sql_delete( 'tuser_task_scheduled', ['id' => $id_console_task] ); if ($result == 1) { return [ 'result' => 0, 'msg' => __('Console Task successfully deleted'), 'id' => false, ]; } // Trick to avoid double execution. header('Location: '.$this->url); } } /** * Disable a recon task. * * @return void */ public function disableTask() { global $config; if (! check_acl($config['id_user'], 0, 'AW')) { db_pandora_audit( AUDIT_LOG_ACL_VIOLATION, 'Trying to access recon task viewer' ); include 'general/noaccess.php'; return; } $task = get_parameter('task', null); if ($task !== null) { $result = db_process_sql_update( 'trecon_task', ['disabled' => 1], ['id_rt' => $task] ); if ($result == 1) { return [ 'result' => 0, 'msg' => __('Task successfully disabled'), 'id' => false, ]; } // Trick to avoid double execution. header('Location: '.$this->url); } } /** * Enable a recon task. * * @return void */ public function enableTask() { global $config; if (! check_acl($config['id_user'], 0, 'AW')) { db_pandora_audit( AUDIT_LOG_ACL_VIOLATION, 'Trying to access recon task viewer' ); include 'general/noaccess.php'; return; } $task = get_parameter('task', null); if ($task !== null) { $result = db_process_sql_update( 'trecon_task', [ 'disabled' => 0, 'status' => 0, ], ['id_rt' => $task] ); if ($result == 1) { return [ 'result' => 0, 'msg' => __('Task successfully enabled'), 'id' => false, ]; } // Trick to avoid double execution. header('Location: '.$this->url); } } /** * Show complete list of running tasks. * * @param string $titleTable Title of section. * @param array $filter Ids array from apps for filter. * @param boolean $extension_section Extension to add in table. * * @return boolean Success or not. */ public function showList($titleTable, $filter, $extension_section=false) { global $config; check_login(); if (!$this->aclMulticheck('AR|AW|AM')) { // Tasklist are allowed only of agent managers. return ''; } // Get all discovery servers. $servers = db_get_all_rows_sql('SELECT * FROM tserver WHERE server_type = 3'); if ($servers === false) { $servers = []; ui_print_error_message(__('Discovery Server is disabled')); $check = db_get_all_rows_sql('SELECT * FROM trecon_task'); return (bool) $check; } else { include_once $config['homedir'].'/include/functions_graph.php'; include_once $config['homedir'].'/include/functions_servers.php'; include_once $config['homedir'].'/include/functions_network_profiles.php'; if (users_is_admin()) { $recon_tasks = db_get_all_rows_sql( sprintf( 'SELECT tasks.*, apps.section AS section, apps.short_name AS short_name FROM trecon_task tasks LEFT JOIN tdiscovery_apps apps ON tasks.id_app = apps.id_app WHERE type IN (%s) OR section = "%s"', implode(',', $filter), $extension_section ) ); } else { $user_groups = implode( ',', array_keys(users_get_groups()) ); $recon_tasks = db_get_all_rows_sql( sprintf( 'SELECT tasks.*, apps.section AS section, apps.short_name AS short_name FROM trecon_task LEFT JOIN tdiscovery_apps apps ON tasks.id_app = apps.id_app WHERE id_group IN (%s) AND (type IN (%s) OR section = "%s")', $user_groups, implode(',', $filter), $extension_section ) ); } // Show network tasks for Recon Server. if ($recon_tasks === false) { $recon_tasks = []; } $url_ajax = $config['homeurl'].'ajax.php'; $table = new StdClass(); $table->cellpadding = 0; $table->cellspacing = 0; $table->width = '100%'; $table->class = 'info_table'; $table->head = []; $table->data = []; $table->align = []; $table->headstyle = []; $table->style = []; $table->style[4] = 'word-break: break-word;'; for ($i = 0; $i < 9; $i++) { $table->headstyle[$i] = 'text-align: left;'; } // Task name. $table->headstyle[1] .= 'min-width: 170px; width: 300px;'; // Server Name. $table->headstyle[2] .= 'min-width: 130px; width: 400px;'; // Name. $table->headstyle[4] .= 'min-width: 100px; width: 350px;'; // Status. $table->headstyle[5] .= 'min-width: 70px; width: 190px;'; // Task type. $table->headstyle[6] .= 'min-width: 190px; width: 200px;'; // Progress. $table->headstyle[7] .= 'min-width: 60px; width: 150px;'; // Updated at. $table->headstyle[8] .= 'min-width: 50px; width: 150px;'; // Operations. $table->headstyle[9] .= 'min-width: 150px; width: 250px;'; if (check_acl($config['id_user'], 0, 'AW')) { $table->head[0] = __('Force'); $table->align[0] = 'left'; } $table->head[1] = __('Task name'); $table->align[1] = 'left'; $table->head[2] = __('Server name'); $table->align[2] = 'left'; $table->head[3] = __('Interval'); $table->align[3] = 'left'; $table->head[4] = __('Network'); $table->align[4] = 'left'; $table->head[5] = __('Status'); $table->align[5] = 'left'; $table->head[6] = __('Task type'); $table->align[6] = 'left'; $table->head[7] = __('Progress'); $table->align[7] = 'left'; $table->head[8] = __('Updated at'); $table->align[8] = 'left'; $table->head[9] = __('Operations'); $table->align[9] = 'left'; foreach ($recon_tasks as $task) { if ($this->aclMulticheck('AR|AW|AM', $task['id_group']) === false) { continue; } $no_operations = false; $data = []; $server_name = servers_get_name($task['id_recon_server']); // By default. $subnet = $task['subnet']; // Exceptions: IPAM. $ipam = false; if ($task['id_recon_script'] != null) { $recon_script_data = db_get_row( 'trecon_script', 'id_recon_script', $task['id_recon_script'] ); if ($recon_script_data !== false) { $recon_script_name = $recon_script_data['name']; if (io_safe_output($recon_script_name) == 'IPAM Recon' && enterprise_installed() ) { $subnet_obj = json_decode($task['macros'], true); $subnet = $subnet_obj['1']['value']; $tipam_task_id = db_get_value( 'id', 'tipam_network', 'id_recon_task', $task['id_rt'] ); $ipam = true; } } } else { $recon_script_data = false; $recon_script_name = false; } if (($task['disabled'] == 0 && $server_name !== '' && (int) $task['type'] !== DISCOVERY_EXTENSION) || ((int) $task['type'] === DISCOVERY_EXTENSION && (int) $task['setup_complete'] === 1) ) { if (check_acl($config['id_user'], 0, 'AW')) { $data[0] = ''; $data[0] .= html_print_image( 'images/force@svg.svg', true, [ 'title' => __('Force'), 'class' => 'main_menu_icon invert_filter', ] ); $data[0] .= ''; } } else if ($task['disabled'] == 2 || ((int) $task['type'] === DISCOVERY_EXTENSION && (int) $task['setup_complete'] === 0) ) { $data[0] = ui_print_help_tip( __('This task has not been completely defined, please edit it'), true ); } else { $data[0] = ''; } // Name task. $data[1] = ''; if ($task['disabled'] != 2) { $data[1] .= ''; } if ($task['disabled'] == 1) { $data[1] .= ''.$task['name'].''; } else { $data[1] .= ''.$task['name'].''; } if ($task['disabled'] != 2) { $data[1] .= ''; } $data[2] = $server_name; if ($task['interval_sweep'] > 0) { $data[3] = human_time_description_raw( $task['interval_sweep'] ); } else { $data[3] = __('Manual'); } if ($task['id_recon_script'] == 0 || $ipam === true) { $data[4] = ui_print_truncate_text($subnet, 50, true, true, true, '[…]'); } else { $data[4] = '-'; } $_rs = $this->getStatusMessage($task); $can_be_reviewed = $_rs['can_be_reviewed']; $data[5] = $_rs['message']; switch ($task['type']) { case DISCOVERY_CLOUD_AZURE_COMPUTE: // Discovery Applications MySQL. $data[6] = html_print_image( 'images/plugins@svg.svg', true, [ 'title' => __('Discovery Cloud Azure Compute'), 'class' => 'main_menu_icon invert_filter', ] ).'  '; $data[6] .= __('Cloud.Azure.Compute'); break; case DISCOVERY_CLOUD_AWS_EC2: // Discovery Applications MySQL. $data[6] = html_print_image( 'images/plugins@svg.svg', true, [ 'title' => __('Discovery Cloud AWS EC2'), 'class' => 'main_menu_icon invert_filter', ] ).'  '; $data[6] .= __('Cloud.AWS.EC2'); break; case DISCOVERY_CLOUD_AWS_RDS: // Discovery Cloud RDS. $data[6] = html_print_image( 'images/cluster@os.svg', true, [ 'title' => __('Discovery Cloud RDS'), 'class' => 'main_menu_icon invert_filter', ] ).'  '; $data[6] .= __('Discovery.Cloud.Aws.RDS'); break; case DISCOVERY_CLOUD_AWS_S3: // Discovery Cloud S3. $data[6] = html_print_image( 'images/cluster@os.svg', true, [ 'title' => __('Discovery Cloud S3'), 'class' => 'main_menu_icon invert_filter', ] ).'  '; $data[6] .= __('Discovery.Cloud.Aws.S3'); break; case DISCOVERY_APP_MYSQL: // Discovery Applications MySQL. $data[6] = html_print_image( 'images/cluster@os.svg', true, [ 'title' => __('Discovery Applications MySQL'), 'class' => 'main_menu_icon invert_filter', ] ).'  '; $data[6] .= __('Discovery.App.MySQL'); break; case DISCOVERY_APP_ORACLE: // Discovery Applications Oracle. $data[6] = html_print_image( 'images/cluster@os.svg', true, [ 'title' => __('Discovery Applications Oracle'), 'class' => 'main_menu_icon invert_filter', ] ).'  '; $data[6] .= __('Discovery.App.Oracle'); break; case DISCOVERY_APP_DB2: // Discovery Applications DB2. $data[6] = html_print_image( 'images/cluster@os.svg', true, [ 'title' => __('Discovery Applications DB2'), 'class' => 'main_menu_icon invert_filter', ] ).'  '; $data[6] .= __('Discovery.App.DB2'); break; case DISCOVERY_DEPLOY_AGENTS: // Internal deployment task. $no_operations = true; $data[6] = html_print_image( 'images/osx-terminal@groups.svg', true, ['title' => __('Agent deployment')] ).'  '; $data[6] .= __('Discovery.Agent.Deployment'); break; case DISCOVERY_APP_MICROSOFT_SQL_SERVER: // Discovery Applications Oracle. $data[6] = html_print_image( 'images/cluster@os.svg', true, [ 'title' => __('Discovery Applications Microsoft SQL Server'), 'class' => 'main_menu_icon invert_filter', ] ).'  '; $data[6] .= __('Discovery.App.Microsoft SQL Server'); break; case DISCOVERY_EXTENSION: // Discovery NetScan. $data[6] = html_print_image( 'images/cluster@os.svg', true, [ 'title' => $task['short_name'], 'class' => 'main_menu_icon invert_filter', ] ).'  '; $data[6] .= $task['short_name']; break; case DISCOVERY_HOSTDEVICES: default: if ($task['id_recon_script'] == 0) { // Discovery NetScan. $data[6] = html_print_image( 'images/cluster@os.svg', true, [ 'title' => __('Discovery NetScan'), 'class' => 'main_menu_icon invert_filter', ] ).'  '; $data[6] .= __('Discovery.NetScan'); } else { // APP or external script recon task. $data[6] = html_print_image( 'images/plugins@svg.svg', true, ['class' => 'main_menu_icon invert_filter'] ).'  '; $data[6] .= $recon_script_name; } break; } if ($task['status'] <= 0 || $task['status'] > 100) { $data[7] = '-'; } else { $data[7] = ui_progress( $task['status'], '100%', 1.2, // Color. '#ececec', // Return. true, // Text. '', // Ajax. [ 'page' => 'godmode/servers/discovery', 'interval' => 10, 'simple' => 1, 'data' => [ 'wiz' => 'tasklist', 'id' => $task['id_rt'], 'method' => 'taskProgress', ], ], '' ); } if ($task['utimestamp'] > 0) { $data[8] = ui_print_timestamp( $task['utimestamp'], true ); } else { $data[8] = __('Not executed yet'); } if (!$no_operations) { if ($task['disabled'] != 2) { $data[9] = ''; if ($can_be_reviewed) { $data[9] .= ''; $data[9] .= html_print_image( 'images/expand.png', true, [ 'title' => __('Review results'), 'class' => 'main_menu_icon invert_filter', ] ); $data[9] .= ''; } $data[9] .= ''; $data[9] .= html_print_image( 'images/details.svg', true, [ 'title' => __('View summary'), 'class' => 'main_menu_icon invert_filter', ] ); $data[9] .= ''; } if ($task['disabled'] != 2 && $task['utimestamp'] > 0 && $task['type'] != DISCOVERY_APP_MYSQL && $task['type'] != DISCOVERY_APP_ORACLE && $task['type'] != DISCOVERY_APP_DB2 && $task['type'] != DISCOVERY_APP_SAP && $task['type'] != DISCOVERY_CLOUD_AWS_RDS && $task['type'] != DISCOVERY_CLOUD_AWS_S3 ) { if (check_acl($config['id_user'], 0, 'MR') && (int) $task['type'] !== 15) { $data[9] .= ''; $data[9] .= html_print_image( 'images/web@groups.svg', true, [ 'title' => __('View map'), 'class' => 'main_menu_icon invert_filter', ] ); $data[9] .= ''; } } if (check_acl( $config['id_user'], $task['id_group'], 'AW' ) ) { if ($ipam === true) { if (empty($tipam_task_id) === false) { $data[9] .= ''.html_print_image( 'images/edit.svg', true, [ 'title' => __('Edit task'), 'class' => 'main_menu_icon invert_filter', ] ).''; $data[9] .= ''.html_print_image( 'images/delete.svg', true, [ 'title' => __('Delete task'), 'class' => 'main_menu_icon invert_filter', ] ).''; } else { $data[9] .= ''.html_print_image( 'images/delete.svg', true, [ 'title' => __('Delete task'), 'class' => 'main_menu_icon invert_filter', ] ).''; } } else { $url_edit = sprintf( 'index.php?sec=gservers&sec2=godmode/servers/discovery&%s&task=%d', $this->getTargetWiz($task, $recon_script_data), $task['id_rt'] ); if ((int) $task['type'] === DISCOVERY_EXTENSION) { $url_edit = sprintf( 'index.php?sec=gservers&sec2=godmode/servers/discovery&wiz=%s&mode=%s&id_task=%s', $task['section'], $task['short_name'], $task['id_rt'], ); } // Check if is a H&D, Cloud or Application or IPAM. $data[9] .= ''.html_print_image( 'images/edit.svg', true, [ 'title' => __('Edit task'), 'class' => 'main_menu_icon invert_filter', ] ).''; $data[9] .= ''.html_print_image( 'images/delete.svg', true, [ 'title' => __('Delete task'), 'class' => 'main_menu_icon invert_filter', ] ).''; } if ($task['disabled'] == 1) { $data[9] .= ''.html_print_image( 'images/lightbulb_off.png', true, [ 'title' => __('enable task'), 'class' => 'filter_none', ] ).''; } else if ($task['disabled'] == 0) { $data[9] .= ''.html_print_image( 'images/lightbulb.png', true, ['title' => __('Disable task')] ).''; } } else { $data[9] = ''; } } else { $data[9] = '-'; } $table->cellclass[][9] = 'table_action_buttons'; // Div neccesary for modal progress task. echo ''; array_push($table->data, $data); } if (empty($table->data)) { $content = '
'.__('Server').' '.$server_name.' '.__('has no discovery tasks assigned').'
'; $return = false; } else { $content = html_print_table($table, true); $return = true; } ui_toggle($content, $titleTable, '', '', false); // Div neccesary for modal map task. echo ''; echo ''; echo ''; echo ''; echo ''; echo ''; unset($table); ui_require_javascript_file('pandora_ui'); ui_require_javascript_file('pandora_taskList'); return $return; } return true; } /** * Show complete list of running tasks. * * @return boolean Success or not. */ public function showListConsoleTask() { return enterprise_hook('tasklist_showListConsoleTask', [$this]); } /** * Return target url sub-string to edit target task. * * @param array $task With all data. * @param array $script With all script data or false if undefined. * * @return string */ public function getTargetWiz($task, $script=false) { if ($script !== false || (int) $task['type'] === DISCOVERY_HOSTDEVICES_CUSTOM ) { switch ($script['type']) { case DISCOVERY_SCRIPT_APP_VMWARE: return 'wiz=app&mode=vmware&page=0'; case DISCOVERY_SCRIPT_IPAM_RECON: return ''; case DISCOVERY_SCRIPT_IPMI_RECON: default: return 'wiz=hd&mode=customnetscan'; case DISCOVERY_SCRIPT_CLOUD_AWS: switch ($task['type']) { case DISCOVERY_CLOUD_AWS_EC2: return 'wiz=cloud&mode=amazonws&ki='.$task['auth_strings'].'&page=1'; case DISCOVERY_CLOUD_AZURE_COMPUTE: return 'wiz=cloud&mode=azure&ki='.$task['auth_strings'].'&sub=compute&page=0'; case DISCOVERY_CLOUD_AWS_S3: return 'wiz=cloud&mode=amazonws&ki='.$task['auth_strings'].'&sub=s3&page=0'; default: return 'wiz=cloud'; } } } switch ($task['type']) { case DISCOVERY_APP_MYSQL: return 'wiz=app&mode=mysql&page=0'; case DISCOVERY_APP_MICROSOFT_SQL_SERVER: return 'wiz=app&mode=MicrosoftSQLServer&page=0'; case DISCOVERY_APP_ORACLE: return 'wiz=app&mode=oracle&page=0'; case DISCOVERY_APP_DB2: return 'wiz=app&mode=DB2&page=0'; case DISCOVERY_CLOUD_AWS: case DISCOVERY_CLOUD_AWS_EC2: return 'wiz=cloud&mode=amazonws&ki='.$task['auth_strings'].'&page=1'; case DISCOVERY_CLOUD_AWS_RDS: return 'wiz=cloud&mode=amazonws&ki='.$task['auth_strings'].'&sub=rds&page=0'; case DISCOVERY_APP_SAP: return 'wiz=app&mode=SAP&page=0'; default: if ($task['description'] == 'console_task') { return 'wiz=ctask'; } else { return 'wiz=hd&mode=netscan'; } break; } } /** * Returns percent of completion of target task. * * @return void */ public function taskProgress() { if (!is_ajax()) { echo json_encode(['error' => true]); return; } $id_task = get_parameter('id', 0); if ($id_task <= 0) { echo json_encode(['error' => true]); return; } $status = db_get_value('status', 'trecon_task', 'id_rt', $id_task); if ($status < 0) { $status = 100; } echo json_encode($status); } /** * Generates charts for progress popup. * * @param array $task Task. * * @return string Charts in HTML. */ private function progressTaskGraph($task) { $result = '
'; $result .= '
'; $result .= ''._('Overall Progress').''; $result .= '
'; $result .= progress_circular_bar( $task['id_rt'], ($task['status'] < 0) ? 100 : $task['status'], 150, 150, '#14524f', '%', '', '#ececec', 0 ); $result .= '
'; if ($task['status'] > 0) { switch ($task['stats']['step']) { case STEP_SCANNING: $str = __('Scanning network'); break; case STEP_CAPABILITIES: $str = __('Checking'); break; case STEP_AFT: $str = __('Finding AFT connectivity'); break; case STEP_TRACEROUTE: $str = __('Finding traceroute connectivity'); break; case STEP_GATEWAY: $str = __('Finding gateway connectivity'); break; case STEP_STATISTICS: $str = __('Searching for devices...'); break; case STEP_APP_SCAN: $str = __('Analyzing application...'); break; case STEP_CUSTOM_QUERIES: $str = __('Executing custom queries...'); break; case STEP_MONITORING: $str = __('Testing modules...'); break; case STEP_PROCESSING: $str = __('Processing results...'); break; default: $str = __('Processing...'); break; } $result .= '
'; $result .= '
'; $result .= ''.$str.' '; if (empty($str) === false) { $result .= $task['stats']['c_network_name']; } $result .= ''; $result .= '
'; $result .= progress_circular_bar( $task['id_rt'].'_detail', $task['stats']['c_network_percent'], 150, 150, '#14524f', '%', '', '#ececec', 0 ); $result .= '
'; } if ($task['review_mode'] == DISCOVERY_REVIEW) { if ($task['status'] <= 0 && empty($task['summary']) === false ) { $result .= ''; $result .= '»'.__('Review'); $result .= ''; } } $result .= '
'; return $result; } /** * Generates a summary table for given task. * * @param array $task Task. * * @return string HTML code. code with summary. */ private function progressTaskSummary($task) { global $config; include_once $config['homedir'].'/include/graphs/functions_d3.php'; if (is_array($task) === false) { return ''; } $output = ''; if (is_array($task['stats']) === false && (int) $task['type'] !== DISCOVERY_EXTENSION) { $task['stats'] = json_decode($task['summary'], true); if (json_last_error() !== JSON_ERROR_NONE) { return $task['summary']; } } if (is_array($task['stats']) || (int) $task['type'] === DISCOVERY_EXTENSION) { $i = 0; $table = new StdClasS(); $table->class = 'databox data'; $table->width = '75%'; $table->styleTable = 'margin: 2em auto 0;border: 1px solid #ddd;background: white;'; $table->rowid = []; $table->data = []; if ($task['review_mode'] == DISCOVERY_RESULTS) { $agents_review = db_get_all_rows_filter( 'tdiscovery_tmp_agents', ['id_rt' => $task['id_rt']] ); $agents = 0; $total = 0; if (is_array($agents_review)) { foreach ($agents_review as $agent) { $data = json_decode(base64_decode($agent['data']), true); if (is_array($data) === false) { continue; } if (is_array($data['agent']) === false) { continue; } // Ensure agent_id really exists. $agent_id = agents_get_agent_id( $data['agent']['nombre'], true ); if ($agent_id > 0) { $agents++; } $total++; } } // Content. $table->data[$i][0] = ''.__('Host&devices total').''; $table->data[$i][1] = ''; $table->data[$i][1] .= $total; $table->data[$i++][1] .= ''; $table->data[$i][0] = ''.__('Agents monitored').''; $table->data[$i][1] = ''; $table->data[$i][1] .= $agents; $table->data[$i++][1] .= ''; $table->data[$i][0] = ''.__('Agents pending').''; $table->data[$i][1] = ''; $table->data[$i][1] .= ($total - $agents); $table->data[$i++][1] .= ''; } else if ((int) $task['type'] === DISCOVERY_EXTENSION) { // Content. $countSummary = 1; if (is_array($task['stats']) === true && count(array_filter(array_keys($task['stats']), 'is_numeric')) === count($task['stats'])) { foreach ($task['stats'] as $key => $summary) { $table->data[$i][0] = ''.__('Summary').' '.$countSummary.''; $table->data[$i][1] = ''; $countSummary++; $i++; if (is_array($summary) === true) { if (empty($summary['summary']) === true && empty($summary['info']) === true) { $table->data[$i][0] = json_encode($summary, JSON_PRETTY_PRINT); $table->data[$i][1] = ''; $i++; continue; } $unknownJson = $summary; foreach ($summary as $k2 => $v) { if (is_array($v) === true) { if ($k2 === 'summary') { foreach ($v as $k3 => $v2) { $table->data[$i][0] = $k3; $table->data[$i][1] = $v2; $i++; } unset($unknownJson[$k2]); } } else { if ($k2 === 'info') { $table->data[$i][0] = $v; $table->data[$i][1] = ''; $i++; unset($unknownJson[$k2]); } } } if (empty($unknownJson) === false) { $table->data[$i][0] = json_encode($unknownJson, JSON_PRETTY_PRINT); $table->data[$i][1] = ''; $i++; } } else { $table->data[$i][0] = $summary; $table->data[$i][1] = ''; $i++; } } } else { $table->data[$i][0] = ''.__('Summary').''; $table->data[$i][1] = ''; $i++; $table->data[$i][0] = $task['summary']; $table->data[$i][1] = ''; $i++; } } else { // Content. if (is_array($task['stats']['summary']) === true) { $table->data[$i][0] = ''.__('Hosts discovered').''; $table->data[$i][1] = ''; $table->data[$i][1] .= $task['stats']['summary']['discovered']; $table->data[$i++][1] .= ''; $table->data[$i][0] = ''.__('Alive').''; $table->data[$i][1] = ''; $table->data[$i][1] .= $task['stats']['summary']['alive']; $table->data[$i++][1] .= ''; $table->data[$i][0] = ''.__('Not alive').''; $table->data[$i][1] = ''; $table->data[$i][1] .= $task['stats']['summary']['not_alive']; $table->data[$i++][1] .= ''; if ($task['type'] == DISCOVERY_HOSTDEVICES) { $table->data[$i][0] = ''.__('Responding SNMP').''; $table->data[$i][1] = ''; $table->data[$i][1] .= $task['stats']['summary']['SNMP']; $table->data[$i++][1] .= ''; $table->data[$i][0] = ''.__('Responding WMI').''; $table->data[$i][1] = ''; $table->data[$i][1] .= $task['stats']['summary']['WMI']; $table->data[$i++][1] .= ''; } } else { $table->data[$i][0] = $task['stats']['summary']; } } $output = '
'.__('Summary').'
'; $output .= html_print_table($table, true).''; } return $output; } /** * Content of modal 'task progress', ajax only. * * @return void */ public function progressTaskDiscovery() { if (!is_ajax()) { return; } $id_task = get_parameter('id', 0); if ($id_task <= 0) { echo json_encode(['error' => true]); return; } $task = db_get_row('trecon_task', 'id_rt', $id_task); $task['stats'] = json_decode($task['summary'], true); $summary = $this->progressTaskSummary($task); $output = ''; // Header information. if ((int) $task['status'] <= 0 && empty($summary)) { if ((int) $task['utimestamp'] !== 0) { $output .= $this->progressTaskGraph($task); } else { $outputMessage = __('This task has never executed'); $output .= ui_print_info_message( $outputMessage, '', true ); } } else if ($task['status'] == 1 || ($task['utimestamp'] == 0 && $task['interval_sweep']) ) { $output .= ui_print_info_message( __('Task queued, please wait.'), '', true ).''; } else { $output .= $this->progressTaskGraph($task); } $output .= $summary; echo json_encode(['html' => $output]); } /** * Get a map of target task. * * @return void */ public function taskShowmap() { global $config; include_once $config['homedir'].'/include/class/NetworkMap.class.php'; $id_task = get_parameter('id', 0); $map = new NetworkMap( [ 'id_task' => $id_task, 'pure' => 1, 'widget' => true, 'map_options' => [ 'map_filter' => [ 'x_offs' => 120, 'node_sep' => 10, ], ], ] ); if (count($map->nodes) <= 1) { // No nodes detected in current task definition. $task = db_get_row('trecon_task', 'id_rt', $id_task); if ((int) $task['type'] === DISCOVERY_CLOUD_GCP_COMPUTE_ENGINE) { ui_print_info_message( __('Please ensure instances or regions are being monitorized and \'scan and general monitoring\' is enabled.') ); } } $map->printMap(); } /** * Shows a modal to review results found by discovery task. * * @return void */ public function showTaskReview() { $id_task = get_parameter('id', 0); if ($id_task <= 0) { ui_print_error_message(__('Invalid task')); return; } $task_data = db_get_all_rows_filter( 'tdiscovery_tmp_agents', ['id_rt' => $id_task] ); $task = db_get_row('trecon_task', 'id_rt', $id_task); $simple_data = []; if (is_array($task_data)) { foreach ($task_data as $agent) { $data = json_decode(base64_decode($agent['data']), true); if (is_array($data) === false) { continue; } if (is_array($data['agent']) === false) { continue; } $id = $data['agent']['nombre']; // Partial. $tmp = [ 'id' => $id, 'name' => $id, 'checked' => $data['agent']['checked'], ]; // Ensure agent_id really exists. $agent_id = agents_get_agent_id($data['agent']['nombre'], true); if ($agent_id > 0) { $tmp['disabled'] = 1; $tmp['agent_id'] = $agent_id; $tmp['checked'] = 1; } // Store. $simple_data[] = $tmp; if (is_array($data['modules'])) { // Alphabetically sort. ksort($data['modules'], (SORT_STRING | SORT_FLAG_CASE)); $simple_data = array_merge( $simple_data, array_reduce( $data['modules'], function ($carry, $item) use ($id, $agent_id) { if (empty($item['name'])) { $item['name'] = $item['nombre']; } if ($item['name'] == 'Host Alive') { return $carry; } if (empty($item['name'])) { $item['name'] = $item['nombre']; } $tmp = [ 'name' => $item['name'], 'id' => $id.'-'.$item['name'], 'pid' => $id, 'checked' => $item['checked'], ]; $agentmodule_id = modules_get_agentmodule_id( io_safe_input($item['name']), $agent_id ); if ($agentmodule_id > 0) { $tmp['disabled'] = 1; $tmp['checked'] = 1; $tmp['module_id'] = $agentmodule_id; } $carry[] = $tmp; return $carry; }, [] ) ); } } } echo '
'; echo $this->progressTaskSummary($task); echo '
'; if (count($simple_data) > 0) { echo '
'; echo ''; echo __('Please select devices to be monitored'); echo '
'; echo ''; echo ''; echo ''; echo ''; echo '
'; echo '
'; echo '
'; echo '
'; echo parent::printTree( 'tree', $simple_data ); echo '
'; } else { echo '
'; echo ''; echo __('No devices found in temporary resources, please re-launch.'); echo ''; echo '
'; } } /** * Processes a review over temporary results found by discovery task. * * @return void */ public function parseTaskReview() { $id_task = get_parameter('id', 0); if ($id_task <= 0) { echo $this->error(__('Invalid task')); return; } $ids = []; $n_agents = 0; $selection = io_safe_output(get_parameter('tree-data-tree', '')); if (empty($selection) === false) { $selection = json_decode($selection, true); $ids = array_reduce( $selection, function ($carry, $item) use (&$n_agents) { // String is agent-module. $fields = explode('-', $item['id']); $agent_name = $fields[0]; $module_name = $fields[1]; if ($module_name === null) { // Do not count if already created. if (db_get_value( 'id_agente', 'tagente', 'nombre', io_safe_input($agent_name) ) === false ) { $n_agents++; } } $carry[] = $item['id']; return $carry; } ); } $task_data = db_get_all_rows_filter( 'tdiscovery_tmp_agents', ['id_rt' => $id_task] ); // License precheck. $license = enterprise_hook('license_get_info'); if (is_array($license) === true && $n_agents > ($license['limit'] - $license['count_enabled']) ) { $limit = ($license['limit'] - $license['count_enabled']); echo json_encode( [ 'error' => __( 'Your selection exceeds the agents available on your license. Limit %d', $limit ), ] ); return; } $summary = []; if (is_array($ids)) { foreach ($task_data as $row) { $data = json_decode(base64_decode($row['data']), true); if (is_array($data)) { // Analize each agent. $agent_name = $data['agent']['nombre']; if (in_array($agent_name, $ids)) { if ($data['agent']['checked'] != 1) { $summary[] = '
  • '.$agent_name.'
  • '; } $data['agent']['checked'] = 1; } else { if ($data['agent']['checked'] == 1) { $summary[] = '
  • '.__('Removed').' '.$agent_name.'
  • '; } $data['agent']['checked'] = 0; } // Modules. if (is_array($data['modules'])) { $n_modules = count($data['modules']); foreach ($data['modules'] as $module_name => $module) { if (in_array($agent_name.'-'.$module_name, $ids)) { if ($data['modules'][$module_name]['checked'] != 1) { $summary[] = '
  • '.$agent_name.' - '.$module_name.'
  • '; } $data['modules'][$module_name]['checked'] = 1; } else { if ($data['modules'][$module_name]['checked'] == 1) { if ($module_name != 'Host Alive') { $summary[] = '
  • '.__('Removed').' '.$agent_name.' - '.$module_name.'
  • '; } } $data['modules'][$module_name]['checked'] = 0; } } } // Update data. db_process_sql_update( 'tdiscovery_tmp_agents', [ 'data' => base64_encode(json_encode($data)), 'review_date' => date('Y-m-d H:i:s'), ], [ 'id_rt' => $id_task, 'label' => $agent_name, ] ); } } } // Schedule execution. db_process_sql_update( 'trecon_task', [ 'utimestamp' => 0, 'status' => 1, 'review_mode' => DISCOVERY_RESULTS, ], ['id_rt' => $id_task] ); if (empty($summary)) { $out .= __('No changes. Re-Scheduled'); } else { $out .= __('Scheduled for creation'); $out .= ''; } echo json_encode( ['result' => $out] ); } /** * Generates task status string for given task. * * @param array $task Discovery task (retrieved from DB). * * @return array Message to be displayed and review status. */ public function getStatusMessage(array $task) { $status = ''; $can_be_reviewed = false; if (empty($task['summary']) === false && $task['summary'] == 'cancelled' ) { $status = __('Cancelled').ui_print_help_tip( __('Server has been restarted while executing this task, please retry.'), true ); } else if ($task['review_mode'] == DISCOVERY_STANDARD) { if ($task['status'] <= 0 && empty($task['summary']) === false ) { if ($task['status'] == -2) { $status = __('Failed'); } else { $status = __('Done'); } } else if ($task['utimestamp'] == 0 && empty($task['summary']) ) { $status = __('Not started'); } else if ($task['utimestamp'] > 0) { $status = __('Done'); } else { $status = __('Pending'); } } else { if ($task['status'] <= 0 && empty($task['summary']) === false ) { $can_be_reviewed = true; $status = ''; $status .= __('Review'); $status .= ''; } else if ($task['utimestamp'] == 0 && empty($task['summary']) ) { $status = __('Not started'); } else { if ($task['review_mode'] == DISCOVERY_RESULTS) { $status = __('Processing'); } else { $status = __('Searching'); } } } return [ 'message' => $status, 'can_be_reviewed' => $can_be_reviewed, ]; } }