'A-A Modules',
3 => 'A-A thresholds',
6 => 'Alerts',
];
/**
* Label set for AP clusters.
*
* @var array
*/
public $APLabels = [
2 => 'A-P Modules',
3 => 'A-P thresholds',
4 => 'A-P module',
5 => 'Critical A-P modules',
6 => 'Alerts',
];
/**
* Variable to store error messages while parsing
* different steps.
*
* @var string
*/
public $errMessages = [];
/**
* Current operation (New | Update).
*
* @var string
*/
public $operation;
/**
* Parent url (for go back forms).
*
* @var string
*/
public $parentUrl;
/**
* Current cluster definition (if any).
*
* @var PandoraFMS\ClusterViewer\Cluster
*/
private $cluster;
/**
* Current cluster agent definition (if any).
*
* @var array
*/
private $agent;
/**
* Builds a Cluster Wizard.
*
* @param string $url Main url.
* @param string $operation Operation (new|update).
*/
public function __construct(string $url, string $operation=null)
{
// Check access.
check_login();
ui_require_css_file('wizard');
ui_require_css_file('discovery');
$this->access = 'AW';
$this->operation = $operation;
$this->url = $url;
$this->parentUrl = $url;
$this->page = (int) get_parameter('page', 0);
$this->id = (int) get_parameter('id', 0);
}
/**
* Run wizard.
*
* @return void
* @throws \Exception On error.
*/
public function run()
{
global $config;
ui_require_css_file('cluster_wizard');
$this->operation = get_parameter('op', '');
$cluster_id = get_parameter('id', '');
$name = get_parameter('name', '');
$this->errMessages = [];
// Cluster initialization. Load.
$load_success = true;
try {
if (empty($cluster_id) === false) {
// Load data.
$this->cluster = new Cluster($cluster_id);
if ($this->cluster->agent()->id_agente() === null) {
$this->errMessages['noagent'] = 'Agent associated to cluster does not exist. Please update to create it.';
}
if ($this->cluster->group()->id_grupo() === null) {
throw new \Exception(
'Group associated to cluster does not exist.'
);
}
} else if (empty($name) === false) {
$cluster_data = Cluster::search(['name' => $name]);
if ($cluster_data !== false) {
$init_failed = true;
$this->page--;
throw new \Exception(
__('Cluster already defined, please use another name.')
);
}
}
} catch (\Exception $e) {
$this->errMessages[] = $e->getMessage();
$load_success = false;
}
if (empty($this->cluster) === true) {
// Empty cluster. Initialize.
$this->cluster = new Cluster();
} else {
// Cluster already exists. Update operation.
$this->operation = 'update';
}
try {
// Check user has grants to edit this cluster.
if ($this->operation !== 'new'
&& (!check_acl(
$config['id_user'],
$this->cluster->group()->id_grupo(),
'AW'
))
) {
// User has no grants to edit this cluster.
throw new \Exception(
'You have no permission to edit this cluster.'
);
}
} catch (\Exception $e) {
$this->errMessages[] = $e->getMessage();
$load_success = false;
}
if ($load_success === true) {
if ($this->cluster->id() === null
&& $this->page > 1
) {
$load_success = false;
$this->errMessages[] = 'Please define this cluster first';
} else {
try {
// Parse results (previous step).
$status = $this->parseForm();
} catch (\Exception $e) {
$this->errMessages[] = $e->getMessage();
if ($this->page > 0) {
$this->page--;
}
}
}
}
// Get form structure (current page).
$form = $this->getForm($load_success);
// Force Cluster calculation. Cluster defined.
$this->cluster->force(false);
View::render(
'cluster/edit',
[
'config' => $config,
'wizard' => $this,
'model' => $this,
'cluster' => $this->cluster,
'form' => $form,
]
);
}
/**
* Retrieve page value.
*
* @return integer Page value.
*/
public function getPage()
{
return $this->page;
}
/**
* Set page to target value.,
*
* @param integer $page New page value.
*
* @return void
*/
public function setPage(int $page)
{
$this->page = $page;
}
/**
* Return current operation.
*
* @return string New or Update.
*/
public function getOperation()
{
return $this->operation;
}
/**
* Retrieve labels.
*
* @return array With labels (breadcrum).
*/
public function getLabels()
{
$labels = $this->labels;
if ($this->cluster->cluster_type() !== null) {
if ($this->cluster->cluster_type() === 'AA') {
$labels = ($this->labels + $this->AALabels);
} else {
$labels = ($this->labels + $this->APLabels);
}
}
return $labels;
}
/**
* Returns a goBack form structure.
*
* @param boolean $main Go to main page.
*
* @return array Form structure.
*/
public function getGoBackForm(?bool $main=false)
{
$url = $this->url;
if ($main === false) {
$page = ($this->page - 1);
if ($page >= 0) {
$extra_url = '';
if ($this->cluster->id() !== null) {
$extra_url = '&id='.$this->cluster->id();
if ($this->cluster->cluster_type() === 'AA') {
// Jump from Alerts back to A-A Thresholds.
if ($page === 5) {
$page = 3;
}
}
}
$url = $this->url.'&op='.$this->operation;
$url .= '&page='.$page.$extra_url;
}
}
$form['form']['action'] = $url;
$form['form']['method'] = 'POST';
$form['form']['id'] = 'go-back-form';
$form['inputs'] = [
[
'arguments' => [
'name' => 'submit',
'label' => __('Go back'),
'type' => 'submit',
'attributes' => [
'icon' => 'back',
'mode' => 'secondary',
],
'return' => true,
],
],
];
return $form;
}
/**
* Parse responses from previous steps.
*
* @return void
* @throws \Exception On error.
*/
private function parseForm()
{
global $config;
// Parse user responses.
if ($this->page <= 0) {
// No previous steps, return OK.
return;
}
if ($this->page === 1) {
/*
*
* PARSE DEFINITION.
*
*/
$name = get_parameter('name', '');
$type = get_parameter('type', null);
$description = get_parameter('description', '');
$id_group = get_parameter('id_group', '');
$server_name = get_parameter('server_name', '');
if ($name === ''
&& $type === null
&& $description === ''
&& $id_group === ''
&& $server_name === ''
) {
if ($this->cluster->id() === null) {
throw new \Exception(
'Please fulfill all required fields.'
);
}
// Default values, show page.
return;
}
if (empty($name) === true
|| empty($type) === true
|| empty($id_group) === true
|| empty($server_name) === true
) {
if (empty($server_name) === true) {
throw new \Exception(
'Please select a valid Prediction server'
);
}
throw new \Exception('Please fulfill all required fields');
}
// Verify cluster type is one from the list.
if (in_array($type, ['AA', 'AP']) === false) {
throw new \Exception('Invalid cluster type selected');
}
if ($this->cluster->id() === null) {
// Create.
// 1. Create agent.
$this->cluster->agent()->alias($name);
$this->cluster->agent()->comentarios($description);
$this->cluster->agent()->intervalo(300);
$this->cluster->agent()->id_grupo($id_group);
$this->cluster->agent()->id_os(CLUSTER_OS_ID);
$this->cluster->agent()->server_name($server_name);
$this->cluster->agent()->modo(1);
$this->cluster->agent()->save();
if ($this->cluster->agent()->id_agente() === false) {
throw new \Exception(
'Failed to create agent: '.$config['dbconnection']->error
);
}
// 2. Create cluster entry.
$this->cluster->name($name);
$this->cluster->cluster_type($type);
$this->cluster->description($description);
$this->cluster->group($id_group);
$this->cluster->id_agent($this->cluster->agent()->id_agente());
$this->cluster->save();
if ($this->cluster->id() === null) {
// Delete agent created in previous step.
\agents_delete_agent($this->cluster->agent()->id());
throw new \Exception(
'Failed to create cluster: '.$config['dbconnection']->error
);
}
// 3. Create cluster module in agent.
$this->cluster->agent()->addModule(
[
'nombre' => io_safe_input('Cluster status'),
'id_modulo' => 5,
'prediction_module' => 5,
'custom_integer_1' => $this->cluster->id(),
'id_tipo_modulo' => 1,
'descripcion' => io_safe_input(
'Cluster status information module'
),
'min_warning' => 1,
'min_critical' => 2,
]
);
} else {
// Update.
$this->cluster->name($name);
$this->cluster->cluster_type($type);
$this->cluster->description($description);
$this->cluster->group($id_group);
$this->cluster->agent()->alias($name);
$this->cluster->agent()->comentarios($description);
$this->cluster->agent()->intervalo(300);
$this->cluster->agent()->id_grupo($id_group);
$this->cluster->agent()->id_os(CLUSTER_OS_ID);
$this->cluster->agent()->server_name($server_name);
$this->cluster->agent()->modo(1);
$this->cluster->agent()->save();
// 2. Re link.
$this->cluster->id_agent($this->cluster->agent()->id_agente());
$this->cluster->save();
// If agent has been deleted, recreate module.
if ($this->errMessages['noagent'] !== null) {
// 3. Create cluster module in agent.
$this->cluster->agent()->addModule(
[
'nombre' => io_safe_input('Cluster status'),
'id_modulo' => 5,
'prediction_module' => 5,
'custom_integer_1' => $this->cluster->id(),
'id_tipo_modulo' => 1,
'descripcion' => io_safe_input(
'Cluster status information module'
),
'min_warning' => 1,
'min_critical' => 2,
]
);
}
unset($this->errMessages['noagent']);
}
return;
}
if ($this->page === 2) {
/*
*
* PARSE MEMBERS.
*
*/
// Parse responses from page 1.
$agents_selected = get_parameter('selected-select-members', null);
if ($agents_selected === null) {
// Direct access.
return;
}
// Clear members. Reparse.
$this->cluster->cleanMembers();
// Remove 'None' field.
if (array_search(0, $agents_selected) === 0) {
unset($agents_selected[0]);
}
if (empty($agents_selected) === true) {
throw new \Exception('No members selected');
}
foreach ($agents_selected as $id_agent) {
$agent = $this->cluster->addMember($id_agent);
\db_pandora_audit(
AUDIT_LOG_AGENT_MANAGEMENT,
'Agent '.io_safe_output($agent->alias()).' added to cluster '.io_safe_output(
$this->cluster->name()
)
);
}
$this->cluster->save();
return;
}
if ($this->page === 3) {
/*
*
* PARSE AA MODULES.
*
*/
$aa_modules = get_parameter('selected-select-aa-modules', null);
if (is_array($aa_modules) === true) {
if ($aa_modules[0] === '0') {
unset($aa_modules[0]);
}
$current = array_keys($this->cluster->getAAModules());
$removed = array_diff($current, $aa_modules);
$changes = false;
foreach ($aa_modules as $m) {
$this->cluster->addAAModule($m);
$changes = true;
}
foreach ($removed as $m) {
$this->cluster->removeAAModule($m);
$changes = true;
}
if ($changes === true) {
$this->cluster->save();
}
}
return;
}
if ($this->page === 4) {
/*
*
* PARSE AA THRESHOLDS
*
*/
$modules = $this->cluster->getAAModules();
$changes = false;
foreach ($modules as $item) {
$value_warning = get_parameter(
'warning-'.md5($item->name()),
null
);
$value_critical = get_parameter(
'critical-'.md5($item->name()),
null
);
if ($value_warning !== null) {
$item->warning_limit($value_warning);
$changes = true;
}
if ($value_critical !== null) {
$item->critical_limit($value_critical);
$changes = true;
}
}
if ($changes === true) {
$this->cluster->save();
}
if ($this->cluster->cluster_type() === 'AA') {
// Force next page '6' (alerts).
$this->page = 6;
}
return;
}
if ($this->page === 5) {
/*
*
* PARSE AP MODULES
*
*/
if ($this->cluster->cluster_type() === 'AA') {
// Direct access. Accessed by URL.
$this->page = 0;
throw new \Exception(
'Unavailable page for this cluster type, please follow this wizard.'
);
}
$ap_modules = get_parameter('selected-select-ap-modules', null);
if (is_array($ap_modules) === true) {
if ($ap_modules[0] === '0') {
unset($ap_modules[0]);
}
$current = array_keys($this->cluster->getAPModules());
$removed = array_diff($current, $ap_modules);
$changes = false;
foreach ($ap_modules as $m) {
$this->cluster->addAPModule($m);
$changes = true;
}
foreach ($removed as $m) {
$this->cluster->removeAPModule($m);
$changes = true;
}
if ($changes === true) {
$this->cluster->save();
}
}
return;
}
if ($this->page === 6) {
/*
*
* PARSE AP MODULES CRITICAL
*
*/
if ($this->cluster->cluster_type() === 'AA') {
// Direct access.
return;
}
$modules = $this->cluster->getAPModules();
$changes = false;
foreach ($modules as $item) {
$value = get_parameter_switch(
'switch-'.md5($item->name()),
null
);
if ($value !== null) {
// Unchecked.
$item->is_critical($value);
$changes = true;
}
}
if ($changes === true) {
$this->cluster->save();
}
return;
}
if ($this->page === 7) {
/*
*
* PARSE ALERTS
*
*/
// There is no need to parse anything. Already managed by alert
// builder.
header('Location: '.$this->url.'&op=view&id='.$this->cluster->id());
}
throw new \Exception('Unexpected error');
}
/**
* Retrieves form estructure for current step.
*
* @param boolean $load_success Load process has been success or not.
*
* @return array Form.
*/
private function getForm(?bool $load_success=true)
{
$form = [];
$final = false;
$extra_url = '';
if ($this->cluster->id() !== null) {
$extra_url = '&id='.$this->cluster->id();
}
$url = $this->url.'&op='.$this->operation;
$target_url .= $url.'&page='.($this->page + 1).$extra_url;
$form['form'] = [
'action' => $target_url,
'method' => 'POST',
'extra' => 'autocomplete="false"',
'id' => 'cluster-edit-'.($this->page + 1),
];
if ($load_success === false && $this->page !== 0) {
return [];
}
if ($this->page === 0) {
/*
*
* Page: Cluster Definition.
*
*/
// Input cluster name.
$form['inputs'][] = [
'label' => ''.__('Cluster name').''.ui_print_help_tip(
__('An agent with the same name of the cluster will be created, as well a special service with the same name'),
true
),
'arguments' => [
'name' => 'name',
'value' => $this->cluster->name(),
'type' => 'text',
'size' => 25,
'required' => true,
],
];
// Input cluster type.
$form['inputs'][] = [
'label' => ''.__('Cluster type').''.ui_print_help_tip(
__('AA is a cluster where all members are working. In AP cluster only master member is working'),
true
),
'arguments' => [
'name' => 'type',
'selected' => $this->cluster->cluster_type(),
'type' => 'select',
'fields' => [
'AA' => __('Active - Active'),
'AP' => __('Active - Pasive'),
],
'required' => true,
],
];
// Input cluster description.
$form['inputs'][] = [
'label' => ''.__('Description').'',
'arguments' => [
'name' => 'description',
'value' => $this->cluster->description(),
'type' => 'text',
'size' => 25,
],
];
// Input Group.
$form['inputs'][] = [
'label' => ''.__('Group').''.ui_print_help_tip(
__('Target cluster agent will be stored under this group'),
true
),
'arguments' => [
'name' => 'id_group',
'returnAllGroup' => false,
'privilege' => $this->access,
'type' => 'select_groups',
'selected' => $this->cluster->group()->id_grupo(),
'return' => true,
'required' => true,
],
];
// Input. Servername.
$form['inputs'][] = [
'label' => ''.__('Prediction server').':'.ui_print_help_tip(
__('You must select a Prediction Server to perform all cluster status calculations'),
true
),
'arguments' => [
'type' => 'select_from_sql',
'sql' => sprintf(
'SELECT name as k, name as v
FROM tserver
WHERE server_type = %d
ORDER BY name',
SERVER_TYPE_PREDICTION
),
'name' => 'server_name',
'selected' => $this->cluster->agent()->server_name(),
'return' => true,
'required' => true,
],
];
} else if ($this->page === 1) {
/*
*
* Page: Cluster members.
*
*/
$all_agents = agents_get_agents(
false,
[
'id_agente',
'alias',
]
);
if ($all_agents === false) {
$all_agents = [];
}
$all_agents = array_reduce(
$all_agents,
function ($carry, $item) {
$carry[$item['id_agente']] = $item['alias'];
return $carry;
},
[]
);
$selected = $this->cluster->getMembers();
$selected = array_reduce(
$selected,
function ($carry, $item) use (&$all_agents) {
$carry[$item->id_agente()] = $item->alias();
unset($all_agents[$item->id_agente()]);
return $carry;
},
[]
);
$form['inputs'][] = [
'arguments' => [
'type' => 'select_multiple_filtered',
'class' => 'w80p mw600px',
'name' => 'members',
'available' => $all_agents,
'selected' => $selected,
'group_filter' => [
'page' => 'operation/cluster/cluster',
'method' => 'getAgentsFromGroup',
'id' => $this->cluster->id(),
],
'texts' => [
'title-left' => 'Available agents',
'title-right' => 'Selected cluster members',
],
],
];
} else if ($this->page === 2) {
/*
*
* Page: A-A modules.
*
*/
$selected = $this->cluster->getAAModules();
$selected = array_reduce(
$selected,
function ($carry, $item) {
$name = io_safe_output($item->name());
$carry[$name] = $name;
return $carry;
},
[]
);
$members = $this->cluster->getMembers();
// Agent ids are stored in array keys.
$members = array_keys($members);
// Get common modules.
$modules = \select_modules_for_agent_group(
// Module group. 0 => All.
0,
// Agents.
$members,
// Show all modules or common ones.
false,
// Return.
false,
// Group by name.
true
);
// Escape html special chars on array keys for select value.
$modules = array_combine(
array_map(
function ($k) {
return htmlspecialchars($k);
},
array_keys($modules)
),
$modules
);
$selected = array_combine(
array_map(
function ($k) {
return htmlspecialchars($k);
},
array_keys($selected)
),
$selected
);
$modules = array_diff_key($modules, $selected);
if ($this->cluster->cluster_type() === 'AP') {
$form['inputs'][] = [
'arguments' => [
'type' => 'select_multiple_filtered',
'class' => 'w80p mw600px',
'name' => 'aa-modules',
'available' => $modules,
'selected' => $selected,
'texts' => [
'title-left' => 'Available modules (common)',
'title-right' => 'Selected active-passive modules',
'filter-item' => 'Filter options by module name',
],
'sections' => [
'filters' => 1,
'item-available-filter' => 1,
'item-selected-filter' => 1,
],
],
];
} else if ($this->cluster->cluster_type() === 'AA') {
$form['inputs'][] = [
'arguments' => [
'type' => 'select_multiple_filtered',
'class' => 'w80p mw600px',
'name' => 'aa-modules',
'available' => $modules,
'selected' => $selected,
'texts' => [
'title-left' => 'Available modules (common)',
'title-right' => 'Selected active-active modules',
'filter-item' => 'Filter options by module name',
],
'sections' => [
'filters' => 1,
'item-available-filter' => 1,
'item-selected-filter' => 1,
],
],
];
}
} else if ($this->page === 3) {
/*
*
* Page: A-A module limits.
*
*/
$aa_modules = $this->cluster->getAAModules();
$inputs = [];
foreach ($aa_modules as $module) {
$inputs[] = [
'block_id' => 'from-to-threshold',
'label' => ''.$module->name().'',
'class' => 'flex-row line w100p',
'direct' => 1,
'block_content' => [
[
'label' => ''.$module->name().'',
],
[
'label' => ''.__('critical if').'',
'arguments' => [
'name' => 'critical-'.md5($module->name()),
'type' => 'number',
'value' => $module->critical_limit(),
'required' => true,
],
],
[
'label' => __('% of balanced modules are down (equal or greater).'),
],
],
];
$inputs[] = [
'block_id' => 'from-to-threshold',
'class' => 'flex-row line w100p',
'direct' => 1,
'block_content' => [
[
'label' => ''.$module->name().'',
],
[
'label' => ''.('warning if').'',
'arguments' => [
'name' => 'warning-'.md5($module->name()),
'type' => 'number',
'value' => $module->warning_limit(),
'required' => true,
],
],
[
'label' => __('% of balanced modules are down (equal or greater).'),
],
],
];
$inputs[] = [
'block_id' => 'from-to-threshold',
'class' => 'flex-row line w100p',
'direct' => 1,
'block_content' => [],
];
}
if ($this->cluster->cluster_type() === 'AP') {
$form['inputs'][] = [
'label' => __(
'Please, set thresholds for all active-passive modules'.ui_print_help_tip(
'If you want your cluster module to be critical when 3 of 6 instances are down, set critical to \'50%\'',
true
)
),
'class' => 'indented',
'block_content' => $inputs,
];
} else if ($this->cluster->cluster_type() === 'AA') {
$form['inputs'][] = [
'label' => __(
'Please, set thresholds for all active-active modules'.ui_print_help_tip(
'If you want your cluster module to be critical when 3 of 6 instances are down, set critical to \'50%\'',
true
)
),
'class' => 'indented',
'block_content' => $inputs,
];
}
} else if ($this->page === 4) {
/*
*
* Page: A-P modules.
*
*/
$selected = $this->cluster->getAPModules();
$aa = $this->cluster->getAAModules();
$selected = array_reduce(
$selected,
function ($carry, $item) {
$name = io_safe_output($item->name());
$carry[$name] = $name;
return $carry;
},
[]
);
$aa = array_reduce(
$aa,
function ($carry, $item) {
$name = io_safe_output($item->name());
$carry[$name] = $name;
return $carry;
},
[]
);
$members = $this->cluster->getMembers();
// Agent ids are stored in array keys.
$members = array_keys($members);
// Get common modules.
$modules = \select_modules_for_agent_group(
// Module group. 0 => All.
0,
// Agents.
$members,
// Show all modules or common ones.
true,
// Return.
false,
// Group by name.
true
);
// Exclude AA modules from available options.
$modules = array_diff_key($modules, $aa);
// Exclude already used from available options.
$modules = array_diff_key($modules, $selected);
$form['inputs'][] = [
'arguments' => [
'type' => 'select_multiple_filtered',
'class' => 'w80p mw600px',
'name' => 'ap-modules',
'available' => $modules,
'selected' => $selected,
'texts' => [
'title-left' => 'Available modules (any)',
'title-right' => 'Selected active-passive modules',
'filter-item' => 'Filter options by module name',
],
'sections' => [
'filters' => 1,
'item-available-filter' => 1,
'item-selected-filter' => 1,
],
],
];
} else if ($this->page === 5) {
/*
*
* Page: A-P critical modules.
*
*/
$ap_modules = $this->cluster->getAPModules();
$inputs = [];
foreach ($ap_modules as $module) {
$inputs[] = [
'label' => $module->name(),
'arguments' => [
'type' => 'switch',
'name' => 'switch-'.md5($module->name()),
'value' => $module->is_critical(),
],
];
}
$form['inputs'][] = [
'label' => __(
'Please, check all active-passive modules critical for this cluster'
).ui_print_help_tip(
__('If a critical balanced module is going to critical status, then cluster will be critical.'),
true
),
'class' => 'indented',
'block_content' => $inputs,
];
} else if ($this->page === 6) {
/*
*
* Page: Alerts.
*
*/
ob_start();
global $config;
$id_agente = $this->cluster->agent()->id_agente();
$dont_display_alert_create_bttn = true;
include_once $config['homedir'].'/godmode/alerts/alert_list.php';
include_once $config['homedir'].'/godmode/alerts/alert_list.builder.php';
// XXX: Please do not use this kind of thing never more.
$hack = ob_get_clean();
// TODO: Alert form.
$form['pre-content'] = $hack;
$final = true;
}
// Submit.
$str = __('Next');
if ($this->cluster->id() !== null) {
$str = __('Update and continue');
}
if ($final === true) {
$str = __('Finish');
}
// Submit button.
$form['submit-external-input'] = [
'name' => 'next',
'label' => $str,
'type' => 'submit',
'attributes' => [
'icon' => 'wand',
'mode' => 'primary',
'form' => 'cluster-edit-'.($this->page + 1),
],
'return' => true,
];
return $form;
}
}