You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
3154 lines
93 KiB
3154 lines
93 KiB
1 year ago
|
<?php
|
||
|
/*
|
||
|
** Zabbix
|
||
|
** Copyright (C) 2001-2023 Zabbix SIA
|
||
|
**
|
||
|
** This program is free software; you can redistribute it and/or modify
|
||
|
** it under the terms of the GNU General Public License as published by
|
||
|
** the Free Software Foundation; either version 2 of the License, or
|
||
|
** (at your option) any later version.
|
||
|
**
|
||
|
** This program is distributed in the hope that it will be useful,
|
||
|
** but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
** GNU General Public License for more details.
|
||
|
**
|
||
|
** You should have received a copy of the GNU General Public License
|
||
|
** along with this program; if not, write to the Free Software
|
||
|
** Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||
|
**/
|
||
|
|
||
|
|
||
|
/**
|
||
|
* Class for importing configuration data.
|
||
|
*/
|
||
|
class CConfigurationImport {
|
||
|
|
||
|
/**
|
||
|
* @var CImportDataAdapter
|
||
|
*/
|
||
|
protected $adapter;
|
||
|
|
||
|
/**
|
||
|
* @var CImportReferencer
|
||
|
*/
|
||
|
protected $referencer;
|
||
|
|
||
|
/**
|
||
|
* @var CImportedObjectContainer
|
||
|
*/
|
||
|
protected $importedObjectContainer;
|
||
|
|
||
|
/**
|
||
|
* @var array
|
||
|
*/
|
||
|
protected $options;
|
||
|
|
||
|
/**
|
||
|
* @var array with data read from source string
|
||
|
*/
|
||
|
protected $data;
|
||
|
|
||
|
/**
|
||
|
* @var array cached data from the adapter
|
||
|
*/
|
||
|
protected $formattedData = [];
|
||
|
|
||
|
/**
|
||
|
* Constructor.
|
||
|
* Source string must be suitable for reader class,
|
||
|
* i.e. if string contains json then reader should be able to read json.
|
||
|
*
|
||
|
* @param array $options import options "createMissing", "updateExisting" and "deleteMissing"
|
||
|
* @param CImportReferencer $referencer class containing all importable objects
|
||
|
* @param CImportedObjectContainer $importedObjectContainer class containing processed host and template IDs
|
||
|
*/
|
||
|
public function __construct(array $options, CImportReferencer $referencer,
|
||
|
CImportedObjectContainer $importedObjectContainer) {
|
||
|
$default_options = [
|
||
|
'template_groups' => ['updateExisting' => false, 'createMissing' => false],
|
||
|
'templates' => ['updateExisting' => false, 'createMissing' => false],
|
||
|
'host_groups' => ['updateExisting' => false, 'createMissing' => false],
|
||
|
'hosts' => ['updateExisting' => false, 'createMissing' => false],
|
||
|
'templateDashboards' => ['updateExisting' => false, 'createMissing' => false, 'deleteMissing' => false],
|
||
|
'templateLinkage' => ['createMissing' => false, 'deleteMissing' => false],
|
||
|
'items' => ['updateExisting' => false, 'createMissing' => false, 'deleteMissing' => false],
|
||
|
'discoveryRules' => ['updateExisting' => false, 'createMissing' => false, 'deleteMissing' => false],
|
||
|
'triggers' => ['updateExisting' => false, 'createMissing' => false, 'deleteMissing' => false],
|
||
|
'graphs' => ['updateExisting' => false, 'createMissing' => false, 'deleteMissing' => false],
|
||
|
'httptests' => ['updateExisting' => false, 'createMissing' => false, 'deleteMissing' => false],
|
||
|
'maps' => ['updateExisting' => false, 'createMissing' => false],
|
||
|
'images' => ['updateExisting' => false, 'createMissing' => false],
|
||
|
'mediaTypes' => ['updateExisting' => false, 'createMissing' => false],
|
||
|
'valueMaps' => ['updateExisting' => false, 'createMissing' => false, 'deleteMissing' => false]
|
||
|
];
|
||
|
|
||
|
$options += $default_options;
|
||
|
foreach ($default_options as $entity => $rules) {
|
||
|
$options[$entity] += $rules;
|
||
|
}
|
||
|
|
||
|
$object_options = (
|
||
|
$options['templateLinkage']['createMissing']
|
||
|
|| $options['templateLinkage']['deleteMissing']
|
||
|
|| $options['items']['updateExisting']
|
||
|
|| $options['items']['createMissing']
|
||
|
|| $options['items']['deleteMissing']
|
||
|
|| $options['discoveryRules']['updateExisting']
|
||
|
|| $options['discoveryRules']['createMissing']
|
||
|
|| $options['discoveryRules']['deleteMissing']
|
||
|
|| $options['triggers']['deleteMissing']
|
||
|
|| $options['graphs']['deleteMissing']
|
||
|
|| $options['httptests']['updateExisting']
|
||
|
|| $options['httptests']['createMissing']
|
||
|
|| $options['httptests']['deleteMissing']
|
||
|
);
|
||
|
|
||
|
$options['process_templates'] = (
|
||
|
!$options['templates']['updateExisting']
|
||
|
&& ($object_options
|
||
|
|| $options['templateDashboards']['updateExisting']
|
||
|
|| $options['templateDashboards']['createMissing']
|
||
|
|| $options['templateDashboards']['deleteMissing']
|
||
|
)
|
||
|
);
|
||
|
$options['process_hosts'] = (!$options['hosts']['updateExisting'] && $object_options);
|
||
|
|
||
|
$this->options = $options;
|
||
|
$this->referencer = $referencer;
|
||
|
$this->importedObjectContainer = $importedObjectContainer;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Import configuration data.
|
||
|
*
|
||
|
* @param CImportDataAdapter $adapter an object to provide access to the imported data
|
||
|
*
|
||
|
* @return bool
|
||
|
*
|
||
|
* @throws Exception
|
||
|
*/
|
||
|
public function import(CImportDataAdapter $adapter): bool {
|
||
|
$this->adapter = $adapter;
|
||
|
|
||
|
// Parse all import for references to resolve them all together with less sql count.
|
||
|
$this->gatherReferences();
|
||
|
|
||
|
$this->processTemplateGroups();
|
||
|
$this->processTemplates();
|
||
|
$this->processHostGroups();
|
||
|
$this->processHosts();
|
||
|
|
||
|
// Delete missing objects from processed hosts and templates.
|
||
|
$this->deleteMissingHttpTests();
|
||
|
$this->deleteMissingTemplateDashboards();
|
||
|
$this->deleteMissingDiscoveryRules();
|
||
|
$this->deleteMissingTriggers();
|
||
|
$this->deleteMissingGraphs();
|
||
|
$this->deleteMissingItems();
|
||
|
|
||
|
// Import objects.
|
||
|
$this->processHttpTests();
|
||
|
$this->processItems();
|
||
|
$this->processTriggers();
|
||
|
$this->processDiscoveryRules();
|
||
|
$this->processGraphs();
|
||
|
$this->processImages();
|
||
|
$this->processMaps();
|
||
|
$this->processTemplateDashboards();
|
||
|
$this->processMediaTypes();
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Parse all import data and collect references to objects.
|
||
|
* For host objects it collects host names, for items - host name and item key, etc.
|
||
|
* Collected references are added and resolved via the $this->referencer object.
|
||
|
*
|
||
|
* @see CImportReferencer
|
||
|
*/
|
||
|
protected function gatherReferences(): void {
|
||
|
$template_groups_refs = [];
|
||
|
$templates_refs = [];
|
||
|
$host_groups_refs = [];
|
||
|
$hosts_refs = [];
|
||
|
$items_refs = [];
|
||
|
$valuemaps_refs = [];
|
||
|
$triggers_refs = [];
|
||
|
$graphs_refs = [];
|
||
|
$iconmaps_refs = [];
|
||
|
$images_refs = [];
|
||
|
$maps_refs = [];
|
||
|
$services_refs = [];
|
||
|
$slas_refs = [];
|
||
|
$users_refs = [];
|
||
|
$actions_refs = [];
|
||
|
$media_types_refs = [];
|
||
|
$template_dashboards_refs = [];
|
||
|
$template_macros_refs = [];
|
||
|
$host_macros_refs = [];
|
||
|
$host_prototype_macros_refs = [];
|
||
|
$proxy_refs = [];
|
||
|
$host_prototypes_refs = [];
|
||
|
$httptests_refs = [];
|
||
|
$httpsteps_refs = [];
|
||
|
|
||
|
foreach ($this->getFormattedTemplateGroups() as $group) {
|
||
|
$template_groups_refs[$group['name']] = ['uuid' => $group['uuid']];
|
||
|
}
|
||
|
|
||
|
foreach ($this->getFormattedTemplates() as $template) {
|
||
|
$templates_refs[$template['host']] = ['uuid' => $template['uuid']];
|
||
|
|
||
|
foreach ($template['groups'] as $group) {
|
||
|
$template_groups_refs += [$group['name'] => []];
|
||
|
}
|
||
|
|
||
|
if (array_key_exists('macros', $template)) {
|
||
|
foreach ($template['macros'] as $macro) {
|
||
|
$template_macros_refs[$template['uuid']][] = $macro['macro'];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($template['templates']) {
|
||
|
foreach ($template['templates'] as $linked_template) {
|
||
|
$templates_refs += [$linked_template['name'] => []];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($this->getFormattedHostGroups() as $group) {
|
||
|
$host_groups_refs[$group['name']] = ['uuid' => $group['uuid']];
|
||
|
}
|
||
|
|
||
|
foreach ($this->getFormattedHosts() as $host) {
|
||
|
$hosts_refs[$host['host']] = [];
|
||
|
|
||
|
foreach ($host['groups'] as $group) {
|
||
|
$host_groups_refs += [$group['name'] => []];
|
||
|
}
|
||
|
|
||
|
if (array_key_exists('macros', $host)) {
|
||
|
foreach ($host['macros'] as $macro) {
|
||
|
$host_macros_refs[$host['host']][] = $macro['macro'];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($host['templates']) {
|
||
|
foreach ($host['templates'] as $linked_template) {
|
||
|
$templates_refs += [$linked_template['name'] => []];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($host['proxy']) {
|
||
|
$proxy_refs[$host['proxy']['name']] = [];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($this->getFormattedItems() as $host => $items) {
|
||
|
foreach ($items as $item) {
|
||
|
$items_refs[$host][$item['key_']] = array_key_exists('uuid', $item)
|
||
|
? ['uuid' => $item['uuid']]
|
||
|
: [];
|
||
|
|
||
|
if ($item['valuemap']) {
|
||
|
$valuemaps_refs[$host][$item['valuemap']['name']] = [];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($this->getFormattedDiscoveryRules() as $host => $discovery_rules) {
|
||
|
foreach ($discovery_rules as $discovery_rule) {
|
||
|
$items_refs[$host][$discovery_rule['key_']] = array_key_exists('uuid', $discovery_rule)
|
||
|
? ['uuid' => $discovery_rule['uuid']]
|
||
|
: [];
|
||
|
|
||
|
foreach ($discovery_rule['item_prototypes'] as $item_prototype) {
|
||
|
$items_refs[$host][$item_prototype['key_']] = array_key_exists('uuid', $item_prototype)
|
||
|
? ['uuid' => $item_prototype['uuid']]
|
||
|
: [];
|
||
|
|
||
|
if (!empty($item_prototype['valuemap'])) {
|
||
|
$valuemaps_refs[$host][$item_prototype['valuemap']['name']] = [];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($discovery_rule['trigger_prototypes'] as $trigger) {
|
||
|
$description = $trigger['description'];
|
||
|
$expression = $trigger['expression'];
|
||
|
$recovery_expression = $trigger['recovery_expression'];
|
||
|
|
||
|
$triggers_refs[$description][$expression][$recovery_expression] = array_key_exists('uuid', $trigger)
|
||
|
? ['uuid' => $trigger['uuid']]
|
||
|
: [];
|
||
|
|
||
|
if (array_key_exists('dependencies', $trigger)) {
|
||
|
foreach ($trigger['dependencies'] as $dependency) {
|
||
|
$name = $dependency['name'];
|
||
|
$expression = $dependency['expression'];
|
||
|
$recovery_expression = $dependency['recovery_expression'];
|
||
|
|
||
|
if (!array_key_exists($name, $triggers_refs)
|
||
|
|| !array_key_exists($expression, $triggers_refs[$name])
|
||
|
|| !array_key_exists($recovery_expression, $triggers_refs[$name][$expression])) {
|
||
|
$triggers_refs[$name][$expression][$recovery_expression] = [];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($discovery_rule['graph_prototypes'] as $graph) {
|
||
|
if ($graph['ymin_item_1']) {
|
||
|
$item_host = $graph['ymin_item_1']['host'];
|
||
|
$item_key = $graph['ymin_item_1']['key'];
|
||
|
|
||
|
if (!array_key_exists($item_host, $items_refs)
|
||
|
|| !array_key_exists($item_key, $items_refs[$item_host])) {
|
||
|
$items_refs[$item_host][$item_key] = [];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($graph['ymax_item_1']) {
|
||
|
$item_host = $graph['ymax_item_1']['host'];
|
||
|
$item_key = $graph['ymax_item_1']['key'];
|
||
|
|
||
|
if (!array_key_exists($item_host, $items_refs)
|
||
|
|| !array_key_exists($item_key, $items_refs[$item_host])) {
|
||
|
$items_refs[$item_host][$item_key] = [];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($graph['gitems'] as $gitem) {
|
||
|
$item_host = $gitem['item']['host'];
|
||
|
$item_key = $gitem['item']['key'];
|
||
|
|
||
|
if (!array_key_exists($item_host, $templates_refs)) {
|
||
|
$hosts_refs[$item_host] = [];
|
||
|
}
|
||
|
|
||
|
if (!array_key_exists($item_host, $items_refs)
|
||
|
|| !array_key_exists($item_key, $items_refs[$item_host])) {
|
||
|
$items_refs[$item_host][$item_key] = [];
|
||
|
}
|
||
|
|
||
|
$graphs_refs[$item_host][$graph['name']] = array_key_exists('uuid', $graph)
|
||
|
? ['uuid' => $graph['uuid']]
|
||
|
: [];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($discovery_rule['host_prototypes'] as $host_prototype) {
|
||
|
if (array_key_exists('uuid', $host_prototype)) {
|
||
|
$host_prototypes_refs['uuid'][$host][$discovery_rule['uuid']][] = $host_prototype['uuid'];
|
||
|
}
|
||
|
else {
|
||
|
$host_prototypes_refs['host'][$host][$discovery_rule['key_']][] = $host_prototype['host'];
|
||
|
}
|
||
|
|
||
|
foreach ($host_prototype['group_links'] as $group_prototype) {
|
||
|
$host_groups_refs += [$group_prototype['group']['name'] => []];
|
||
|
}
|
||
|
|
||
|
if (array_key_exists('macros', $host_prototype)) {
|
||
|
foreach ($host_prototype['macros'] as $macro) {
|
||
|
if (array_key_exists('uuid', $host_prototype)) {
|
||
|
$host_prototype_macros_refs['uuid'][$host][$discovery_rule['key_']]
|
||
|
[$host_prototype['uuid']][] = $macro['macro'];
|
||
|
}
|
||
|
else {
|
||
|
$host_prototype_macros_refs['host'][$host][$discovery_rule['key_']]
|
||
|
[$host_prototype['host']][] = $macro['macro'];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($host_prototype['templates'] as $template) {
|
||
|
$templates_refs += [$template['name'] => []];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($discovery_rule['overrides']) {
|
||
|
foreach ($discovery_rule['overrides'] as $override) {
|
||
|
foreach ($override['operations'] as $operation) {
|
||
|
if ($operation['operationobject'] == OPERATION_OBJECT_HOST_PROTOTYPE
|
||
|
&& array_key_exists('optemplate', $operation)) {
|
||
|
foreach ($operation['optemplate'] as $template) {
|
||
|
$templates_refs += [$template['name'] => []];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($this->getFormattedGraphs() as $graph) {
|
||
|
if ($graph['ymin_item_1']) {
|
||
|
$item_host = $graph['ymin_item_1']['host'];
|
||
|
$item_key = $graph['ymin_item_1']['key'];
|
||
|
|
||
|
if (!array_key_exists($item_host, $templates_refs)) {
|
||
|
$hosts_refs[$item_host] = [];
|
||
|
}
|
||
|
|
||
|
if (!array_key_exists($item_host, $items_refs)
|
||
|
|| !array_key_exists($item_key, $items_refs[$item_host])) {
|
||
|
$items_refs[$item_host][$item_key] = [];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($graph['ymax_item_1']) {
|
||
|
$item_host = $graph['ymax_item_1']['host'];
|
||
|
$item_key = $graph['ymax_item_1']['key'];
|
||
|
|
||
|
if (!array_key_exists($item_host, $templates_refs)) {
|
||
|
$hosts_refs[$item_host] = [];
|
||
|
}
|
||
|
|
||
|
if (!array_key_exists($item_host, $items_refs)
|
||
|
|| !array_key_exists($item_key, $items_refs[$item_host])) {
|
||
|
$items_refs[$item_host][$item_key] = [];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (array_key_exists('gitems', $graph) && $graph['gitems']) {
|
||
|
foreach ($graph['gitems'] as $gitem) {
|
||
|
$item_host = $gitem['item']['host'];
|
||
|
$item_key = $gitem['item']['key'];
|
||
|
|
||
|
if (!array_key_exists($item_host, $templates_refs)) {
|
||
|
$hosts_refs[$item_host] = [];
|
||
|
}
|
||
|
|
||
|
if (!array_key_exists($item_host, $items_refs)
|
||
|
|| !array_key_exists($item_key, $items_refs[$item_host])) {
|
||
|
$items_refs[$item_host][$item_key] = [];
|
||
|
}
|
||
|
|
||
|
$graphs_refs[$gitem['item']['host']][$graph['name']] = array_key_exists('uuid', $graph)
|
||
|
? ['uuid' => $graph['uuid']]
|
||
|
: [];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($this->getFormattedTriggers() as $trigger) {
|
||
|
$triggers_refs[$trigger['description']][$trigger['expression']][$trigger['recovery_expression']] =
|
||
|
array_key_exists('uuid', $trigger)
|
||
|
? ['uuid' => $trigger['uuid']]
|
||
|
: [];
|
||
|
|
||
|
if (array_key_exists('dependencies', $trigger)) {
|
||
|
foreach ($trigger['dependencies'] as $dependency) {
|
||
|
$name = $dependency['name'];
|
||
|
$expression = $dependency['expression'];
|
||
|
$recovery_expression = $dependency['recovery_expression'];
|
||
|
|
||
|
if (!array_key_exists($name, $triggers_refs)
|
||
|
|| !array_key_exists($expression, $triggers_refs[$name])
|
||
|
|| !array_key_exists($recovery_expression, $triggers_refs[$name][$expression])) {
|
||
|
$triggers_refs[$name][$expression][$recovery_expression] = [];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($this->getFormattedMaps() as $map) {
|
||
|
$maps_refs[$map['name']] = [];
|
||
|
|
||
|
if ($map['iconmap'] && array_key_exists('name', $map['iconmap']) && $map['iconmap']['name'] !== '') {
|
||
|
$iconmaps_refs[$map['iconmap']['name']] = [];
|
||
|
}
|
||
|
|
||
|
if ($map['background'] && array_key_exists('name', $map['background'])
|
||
|
&& $map['background']['name'] !== '') {
|
||
|
$images_refs[$map['background']['name']] = [];
|
||
|
}
|
||
|
|
||
|
if (array_key_exists('selements', $map)) {
|
||
|
foreach ($map['selements'] as $selement) {
|
||
|
switch ($selement['elementtype']) {
|
||
|
case SYSMAP_ELEMENT_TYPE_MAP:
|
||
|
$maps_refs[$selement['elements'][0]['name']] = [];
|
||
|
break;
|
||
|
|
||
|
case SYSMAP_ELEMENT_TYPE_HOST_GROUP:
|
||
|
$host_groups_refs += [$selement['elements'][0]['name'] => []];
|
||
|
break;
|
||
|
|
||
|
case SYSMAP_ELEMENT_TYPE_HOST:
|
||
|
$hosts_refs += [$selement['elements'][0]['host'] => []];
|
||
|
break;
|
||
|
|
||
|
case SYSMAP_ELEMENT_TYPE_TRIGGER:
|
||
|
foreach ($selement['elements'] as $element) {
|
||
|
$description = $element['description'];
|
||
|
$expression = $element['expression'];
|
||
|
$recovery_expression = $element['recovery_expression'];
|
||
|
|
||
|
if (!array_key_exists($description, $triggers_refs)
|
||
|
|| !array_key_exists($expression, $triggers_refs[$description])
|
||
|
|| !array_key_exists($recovery_expression,
|
||
|
$triggers_refs[$description][$expression])) {
|
||
|
$triggers_refs[$description][$expression][$recovery_expression] = [];
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (array_key_exists('links', $map)) {
|
||
|
foreach ($map['links'] as $link) {
|
||
|
if (array_key_exists('linktriggers', $link)) {
|
||
|
foreach ($link['linktriggers'] as $link_trigger) {
|
||
|
$description = $link_trigger['trigger']['description'];
|
||
|
$expression = $link_trigger['trigger']['expression'];
|
||
|
$recovery_expression = $link_trigger['trigger']['recovery_expression'];
|
||
|
|
||
|
if (!array_key_exists($description, $triggers_refs)
|
||
|
|| !array_key_exists($expression, $triggers_refs[$description])
|
||
|
|| !array_key_exists($recovery_expression,
|
||
|
$triggers_refs[$description][$expression])) {
|
||
|
$triggers_refs[$description][$expression][$recovery_expression] = [];
|
||
|
}
|
||
|
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($this->getFormattedTemplateDashboards() as $host => $dashboards) {
|
||
|
foreach ($dashboards as $dashboard) {
|
||
|
$template_dashboards_refs[$dashboard['uuid']]['name'] = $dashboard['name'];
|
||
|
|
||
|
if (!$dashboard['pages']) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
foreach ($dashboard['pages'] as $dashboard_page) {
|
||
|
if (!$dashboard_page['widgets']) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
foreach ($dashboard_page['widgets'] as $widget) {
|
||
|
foreach ($widget['fields'] as $field) {
|
||
|
$value = $field['value'];
|
||
|
|
||
|
switch ($field['type']) {
|
||
|
case ZBX_WIDGET_FIELD_TYPE_ITEM:
|
||
|
case ZBX_WIDGET_FIELD_TYPE_ITEM_PROTOTYPE:
|
||
|
$templates_refs += [$value['host'] => []];
|
||
|
|
||
|
if (!array_key_exists($value['host'], $items_refs)
|
||
|
|| !array_key_exists($value['key'], $items_refs[$value['host']])) {
|
||
|
$items_refs[$value['host']][$value['key']] = [];
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case ZBX_WIDGET_FIELD_TYPE_GRAPH:
|
||
|
case ZBX_WIDGET_FIELD_TYPE_GRAPH_PROTOTYPE:
|
||
|
$templates_refs += [$value['host'] => []];
|
||
|
|
||
|
if (!array_key_exists($value['host'], $graphs_refs)
|
||
|
|| !array_key_exists($value['name'], $graphs_refs[$value['host']])) {
|
||
|
$graphs_refs[$value['host']][$value['name']] = [];
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case ZBX_WIDGET_FIELD_TYPE_MAP:
|
||
|
if (!array_key_exists($value['name'], $maps_refs)) {
|
||
|
$maps_refs[$value['name']] = [];
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case ZBX_WIDGET_FIELD_TYPE_SERVICE:
|
||
|
if (!array_key_exists($value['name'], $services_refs)) {
|
||
|
$services_refs[$value['name']] = [];
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case ZBX_WIDGET_FIELD_TYPE_SLA:
|
||
|
if (!array_key_exists($value['name'], $slas_refs)) {
|
||
|
$slas_refs[$value['name']] = [];
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case ZBX_WIDGET_FIELD_TYPE_USER:
|
||
|
if (!array_key_exists($value['username'], $users_refs)) {
|
||
|
$users_refs[$value['username']] = [];
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case ZBX_WIDGET_FIELD_TYPE_ACTION:
|
||
|
if (!array_key_exists($value['name'], $actions_refs)) {
|
||
|
$actions_refs[$value['name']] = [];
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case ZBX_WIDGET_FIELD_TYPE_MEDIA_TYPE:
|
||
|
if (!array_key_exists($value['name'], $media_types_refs)) {
|
||
|
$media_types_refs[$value['name']] = [];
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($this->getFormattedHttpTests() as $host => $httptests) {
|
||
|
foreach ($httptests as $httptest) {
|
||
|
$httptests_refs[$host][$httptest['name']] = array_key_exists('uuid', $httptest)
|
||
|
? ['uuid' => $httptest['uuid']]
|
||
|
: [];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($this->getFormattedHttpSteps() as $host => $httptests) {
|
||
|
foreach ($httptests as $httptest_name => $httpsteps) {
|
||
|
foreach ($httpsteps as $httpstep) {
|
||
|
$httpsteps_refs[$host][$httptest_name][$httpstep['name']] = [];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($this->getFormattedImages() as $image) {
|
||
|
$images_refs[$image['name']] = [];
|
||
|
}
|
||
|
|
||
|
$this->referencer->addTemplateGroups($template_groups_refs);
|
||
|
$this->referencer->addTemplates($templates_refs);
|
||
|
$this->referencer->addHostGroups($host_groups_refs);
|
||
|
$this->referencer->addHosts($hosts_refs);
|
||
|
$this->referencer->addItems($items_refs);
|
||
|
$this->referencer->addValuemaps($valuemaps_refs);
|
||
|
$this->referencer->addTriggers($triggers_refs);
|
||
|
$this->referencer->addGraphs($graphs_refs);
|
||
|
$this->referencer->addIconmaps($iconmaps_refs);
|
||
|
$this->referencer->addImages($images_refs);
|
||
|
$this->referencer->addMaps($maps_refs);
|
||
|
$this->referencer->addServices($services_refs);
|
||
|
$this->referencer->addSlas($slas_refs);
|
||
|
$this->referencer->addUsers($users_refs);
|
||
|
$this->referencer->addActions($actions_refs);
|
||
|
$this->referencer->addMediaTypes($media_types_refs);
|
||
|
$this->referencer->addTemplateDashboards($template_dashboards_refs);
|
||
|
$this->referencer->addTemplateMacros($template_macros_refs);
|
||
|
$this->referencer->addHostMacros($host_macros_refs);
|
||
|
$this->referencer->addHostPrototypeMacros($host_prototype_macros_refs);
|
||
|
$this->referencer->addProxies($proxy_refs);
|
||
|
$this->referencer->addHostPrototypes($host_prototypes_refs);
|
||
|
$this->referencer->addHttpTests($httptests_refs);
|
||
|
$this->referencer->addHttpSteps($httpsteps_refs);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Import template groups.
|
||
|
*/
|
||
|
protected function processTemplateGroups(): void {
|
||
|
if (!$this->options['template_groups']['createMissing']
|
||
|
&& !$this->options['template_groups']['updateExisting']) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$groups_to_create = [];
|
||
|
$groups_to_update = [];
|
||
|
|
||
|
foreach ($this->getFormattedTemplateGroups() as $group) {
|
||
|
$groupid = $this->referencer->findTemplateGroupidByUuid($group['uuid']);
|
||
|
|
||
|
if ($groupid === null) {
|
||
|
$groupid = $this->referencer->findTemplateGroupidByName($group['name']);
|
||
|
}
|
||
|
|
||
|
if ($groupid !== null) {
|
||
|
$groups_to_update[] = $group + ['groupid' => $groupid];
|
||
|
}
|
||
|
else {
|
||
|
$groups_to_create[] = $group;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($this->options['template_groups']['updateExisting'] && $groups_to_update) {
|
||
|
API::TemplateGroup()->update($groups_to_update);
|
||
|
|
||
|
foreach ($groups_to_update as $group) {
|
||
|
$this->referencer->setDbTemplateGroup($group['groupid'], $group);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($this->options['template_groups']['createMissing'] && $groups_to_create) {
|
||
|
$created_groups = API::TemplateGroup()->create($groups_to_create);
|
||
|
|
||
|
foreach ($created_groups['groupids'] as $index => $groupid) {
|
||
|
$this->referencer->setDbTemplateGroup($groupid, $groups_to_create[$index]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Import host groups.
|
||
|
*/
|
||
|
protected function processHostGroups(): void {
|
||
|
if (!$this->options['host_groups']['createMissing'] && !$this->options['host_groups']['updateExisting']) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$groups_to_create = [];
|
||
|
$groups_to_update = [];
|
||
|
|
||
|
foreach ($this->getFormattedHostGroups() as $group) {
|
||
|
$groupid = $this->referencer->findHostGroupidByUuid($group['uuid']);
|
||
|
|
||
|
if ($groupid === null) {
|
||
|
$groupid = $this->referencer->findHostGroupidByName($group['name']);
|
||
|
}
|
||
|
|
||
|
if ($groupid !== null) {
|
||
|
$groups_to_update[] = $group + ['groupid' => $groupid];
|
||
|
}
|
||
|
else {
|
||
|
$groups_to_create[] = $group;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($this->options['host_groups']['updateExisting'] && $groups_to_update) {
|
||
|
API::HostGroup()->update($groups_to_update);
|
||
|
|
||
|
foreach ($groups_to_update as $group) {
|
||
|
$this->referencer->setDbHostGroup($group['groupid'], $group);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($this->options['host_groups']['createMissing'] && $groups_to_create) {
|
||
|
$created_groups = API::HostGroup()->create($groups_to_create);
|
||
|
|
||
|
foreach ($created_groups['groupids'] as $index => $groupid) {
|
||
|
$this->referencer->setDbHostGroup($groupid, $groups_to_create[$index]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Import templates.
|
||
|
*
|
||
|
* @throws Exception
|
||
|
*/
|
||
|
protected function processTemplates(): void {
|
||
|
if ($this->options['templates']['updateExisting'] || $this->options['templates']['createMissing']
|
||
|
|| $this->options['process_templates']) {
|
||
|
$templates = $this->getFormattedTemplates();
|
||
|
|
||
|
if ($templates) {
|
||
|
$template_importer = new CTemplateImporter($this->options, $this->referencer,
|
||
|
$this->importedObjectContainer
|
||
|
);
|
||
|
$template_importer->import($templates);
|
||
|
|
||
|
// Get list of imported template IDs and add them processed template ID list.
|
||
|
$templateids = $template_importer->getProcessedTemplateids();
|
||
|
$this->importedObjectContainer->addTemplateIds($templateids);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Import hosts.
|
||
|
*
|
||
|
* @throws Exception
|
||
|
*/
|
||
|
protected function processHosts(): void {
|
||
|
if ($this->options['hosts']['updateExisting'] || $this->options['hosts']['createMissing']
|
||
|
|| $this->options['process_hosts']) {
|
||
|
$hosts = $this->getFormattedHosts();
|
||
|
|
||
|
if ($hosts) {
|
||
|
$host_importer = new CHostImporter($this->options, $this->referencer, $this->importedObjectContainer);
|
||
|
$host_importer->import($hosts);
|
||
|
|
||
|
// Get list of imported host IDs and add them processed host ID list.
|
||
|
$hostids = $host_importer->getProcessedHostIds();
|
||
|
$this->importedObjectContainer->addHostIds($hostids);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Import items.
|
||
|
*
|
||
|
* @throws Exception
|
||
|
*/
|
||
|
protected function processItems(): void {
|
||
|
if (!$this->options['items']['createMissing'] && !$this->options['items']['updateExisting']) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$master_item_key = 'master_item';
|
||
|
$order_tree = $this->getItemsOrder($master_item_key);
|
||
|
|
||
|
$items_to_create = [];
|
||
|
$items_to_update = [];
|
||
|
$levels = [];
|
||
|
|
||
|
foreach ($this->getFormattedItems() as $host => $items) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($host);
|
||
|
|
||
|
if ($hostid === null
|
||
|
|| (!$this->importedObjectContainer->isHostProcessed($hostid)
|
||
|
&& !$this->importedObjectContainer->isTemplateProcessed($hostid))) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
foreach ($order_tree[$host] as $item_key => $level) {
|
||
|
$item = $items[$item_key];
|
||
|
$item['hostid'] = $hostid;
|
||
|
unset($item['triggers']);
|
||
|
|
||
|
$levels[$level] = true;
|
||
|
|
||
|
$delay_types = [ITEM_TYPE_ZABBIX, ITEM_TYPE_SIMPLE, ITEM_TYPE_INTERNAL, ITEM_TYPE_ZABBIX_ACTIVE,
|
||
|
ITEM_TYPE_EXTERNAL, ITEM_TYPE_DB_MONITOR, ITEM_TYPE_IPMI, ITEM_TYPE_SSH, ITEM_TYPE_TELNET,
|
||
|
ITEM_TYPE_CALCULATED, ITEM_TYPE_JMX, ITEM_TYPE_HTTPAGENT, ITEM_TYPE_SNMP, ITEM_TYPE_SCRIPT
|
||
|
];
|
||
|
|
||
|
if (!in_array($item['type'], $delay_types)) {
|
||
|
unset($item['delay']);
|
||
|
}
|
||
|
|
||
|
if (array_key_exists('interface_ref', $item) && $item['interface_ref']) {
|
||
|
$interfaceid = $this->referencer->findInterfaceidByRef($hostid, $item['interface_ref']);
|
||
|
|
||
|
if ($interfaceid === null) {
|
||
|
throw new Exception(_s('Cannot find interface "%1$s" used for item "%2$s" on "%3$s".',
|
||
|
$item['interface_ref'], $item['name'], $host
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$item['interfaceid'] = $interfaceid;
|
||
|
}
|
||
|
unset($item['interface_ref']);
|
||
|
|
||
|
if (array_key_exists('valuemap', $item) && $item['valuemap']) {
|
||
|
$valuemapid = $this->referencer->findValuemapidByName($hostid, $item['valuemap']['name']);
|
||
|
|
||
|
if ($valuemapid === null) {
|
||
|
throw new Exception(_s(
|
||
|
'Cannot find value map "%1$s" used for item "%2$s" on "%3$s".',
|
||
|
$item['valuemap']['name'],
|
||
|
$item['name'],
|
||
|
$host
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$item['valuemapid'] = $valuemapid;
|
||
|
}
|
||
|
unset($item['valuemap']);
|
||
|
|
||
|
if ($item['type'] == ITEM_TYPE_DEPENDENT) {
|
||
|
if (!array_key_exists('key', $item[$master_item_key])) {
|
||
|
throw new Exception(_s('Incorrect value for field "%1$s": %2$s.', 'master_itemid',
|
||
|
_('cannot be empty')
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$master_itemid = $this->referencer->findItemidByKey($hostid, $item[$master_item_key]['key'], true);
|
||
|
|
||
|
if ($master_itemid !== null) {
|
||
|
$item['master_itemid'] = $master_itemid;
|
||
|
unset($item[$master_item_key]);
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
unset($item[$master_item_key]);
|
||
|
}
|
||
|
|
||
|
if ($item['type'] == ITEM_TYPE_HTTPAGENT) {
|
||
|
$headers = [];
|
||
|
|
||
|
foreach ($item['headers'] as $header) {
|
||
|
$headers[$header['name']] = $header['value'];
|
||
|
}
|
||
|
|
||
|
$item['headers'] = $headers;
|
||
|
|
||
|
$query_fields = [];
|
||
|
|
||
|
foreach ($item['query_fields'] as $query_field) {
|
||
|
$query_fields[] = [$query_field['name'] => $query_field['value']];
|
||
|
}
|
||
|
|
||
|
$item['query_fields'] = $query_fields;
|
||
|
}
|
||
|
|
||
|
foreach ($item['preprocessing'] as &$preprocessing_step) {
|
||
|
$preprocessing_step['params'] = implode("\n", $preprocessing_step['parameters']);
|
||
|
|
||
|
unset($preprocessing_step['parameters']);
|
||
|
}
|
||
|
unset($preprocessing_step);
|
||
|
|
||
|
$itemid = null;
|
||
|
|
||
|
if (array_key_exists('uuid', $item)) {
|
||
|
$itemid = $this->referencer->findItemidByUuid($item['uuid']);
|
||
|
}
|
||
|
|
||
|
if ($itemid === null) {
|
||
|
$itemid = $this->referencer->findItemidByKey($hostid, $item['key_']);
|
||
|
}
|
||
|
|
||
|
if ($itemid !== null) {
|
||
|
$item['itemid'] = $itemid;
|
||
|
|
||
|
$items_to_update[$level][] = $item;
|
||
|
}
|
||
|
else {
|
||
|
$items_to_create[$level][] = $item;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ksort($levels);
|
||
|
foreach (array_keys($levels) as $level) {
|
||
|
if ($this->options['items']['updateExisting'] && array_key_exists($level, $items_to_update)) {
|
||
|
$this->updateItemsWithDependency([$items_to_update[$level]], $master_item_key, API::Item());
|
||
|
}
|
||
|
if ($this->options['items']['createMissing'] && array_key_exists($level, $items_to_create)) {
|
||
|
$this->createItemsWithDependency([$items_to_create[$level]], $master_item_key, API::Item());
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Refresh items because templated ones can be inherited to host and used in triggers, graphs, etc.
|
||
|
$this->referencer->refreshItems();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Create CItem or CItemPrototype with dependency.
|
||
|
*
|
||
|
* @param array $items_by_level Associative array of entities where key is entity dependency
|
||
|
* level and value is array of entities for this level.
|
||
|
* @param string $master_item_key Master entity array key in xml parsed data.
|
||
|
* @param CItem|CItemPrototype $api_service Entity service which is capable to proceed with entity create.
|
||
|
*
|
||
|
* @throws Exception if entity master entity can not be resolved.
|
||
|
*/
|
||
|
protected function createItemsWithDependency(array $items_by_level, string $master_item_key,
|
||
|
CItemGeneral $api_service): void {
|
||
|
foreach ($items_by_level as $items_to_create) {
|
||
|
foreach ($items_to_create as &$item) {
|
||
|
if (array_key_exists($master_item_key, $item)) {
|
||
|
$item['master_itemid'] = $this->referencer->findItemidByKey($item['hostid'],
|
||
|
$item[$master_item_key]['key'], true
|
||
|
);
|
||
|
|
||
|
if ($item['master_itemid'] === null) {
|
||
|
throw new Exception(_s('Incorrect value for field "%1$s": %2$s.',
|
||
|
'master_itemid', _s('value "%1$s" not found', $item[$master_item_key]['key'])
|
||
|
));
|
||
|
}
|
||
|
unset($item[$master_item_key]);
|
||
|
}
|
||
|
}
|
||
|
unset($item);
|
||
|
|
||
|
$created_items = $api_service->create($items_to_create);
|
||
|
|
||
|
foreach ($items_to_create as $index => $item) {
|
||
|
$this->referencer->setDbItem($created_items['itemids'][$index], $item);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Update CItem or CItemPrototype with dependency.
|
||
|
*
|
||
|
* @param array $items_by_level Associative array of entities where key is entity dependency
|
||
|
* level and value is array of entities for this level.
|
||
|
* @param string $master_item_key Master entity array key in xml parsed data.
|
||
|
* @param CItem|CItemPrototype $api_service Entity service which is capable to proceed with entity update.
|
||
|
*
|
||
|
* @throws Exception if entity master entity can not be resolved.
|
||
|
*/
|
||
|
protected function updateItemsWithDependency(array $items_by_level, string $master_item_key,
|
||
|
CItemGeneral $api_service): void {
|
||
|
foreach ($items_by_level as $items_to_update) {
|
||
|
$hostids = [];
|
||
|
|
||
|
foreach ($items_to_update as &$item) {
|
||
|
if (array_key_exists($master_item_key, $item)) {
|
||
|
$item['master_itemid'] = $this->referencer->findItemidByKey($item['hostid'],
|
||
|
$item[$master_item_key]['key'], true
|
||
|
);
|
||
|
|
||
|
if ($item['master_itemid'] === null) {
|
||
|
throw new Exception(_s('Incorrect value for field "%1$s": %2$s.',
|
||
|
'master_itemid', _s('value "%1$s" not found', $item[$master_item_key]['key'])
|
||
|
));
|
||
|
}
|
||
|
unset($item[$master_item_key]);
|
||
|
}
|
||
|
|
||
|
$hostids[] = $item['hostid'];
|
||
|
unset($item['hostid']);
|
||
|
}
|
||
|
unset($item);
|
||
|
|
||
|
$updated_items = $api_service->update($items_to_update);
|
||
|
|
||
|
foreach ($items_to_update as $index => $item) {
|
||
|
$item['hostid'] = array_shift($hostids);
|
||
|
|
||
|
$this->referencer->setDbItem($updated_items['itemids'][$index], $item);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Import discovery rules.
|
||
|
*
|
||
|
* @throws Exception
|
||
|
*/
|
||
|
protected function processDiscoveryRules(): void {
|
||
|
if (!$this->options['discoveryRules']['createMissing'] && !$this->options['discoveryRules']['updateExisting']) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$master_item_key = 'master_item';
|
||
|
$discovery_rules_by_hosts = $this->getFormattedDiscoveryRules();
|
||
|
|
||
|
if (!$discovery_rules_by_hosts) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Unset rules that are related to hosts we did not process.
|
||
|
foreach ($discovery_rules_by_hosts as $host => $discovery_rules) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($host);
|
||
|
|
||
|
if ($hostid === null
|
||
|
|| (!$this->importedObjectContainer->isHostProcessed($hostid)
|
||
|
&& !$this->importedObjectContainer->isTemplateProcessed($hostid))) {
|
||
|
unset($discovery_rules_by_hosts[$host]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($this->options['discoveryRules']['updateExisting']) {
|
||
|
$this->deleteMissingPrototypes($discovery_rules_by_hosts);
|
||
|
}
|
||
|
|
||
|
$discovery_rules_to_create = [];
|
||
|
$discovery_rules_to_update = [];
|
||
|
$upd_discovery_rule_hostids = [];
|
||
|
|
||
|
/*
|
||
|
* It's possible that some LLD rules use master items which are web items. They don't reside in item
|
||
|
* references at this point. For items and item prototypes web items are found while processing the order of
|
||
|
* them, but for LLD rules there is no ordering, so this is done independently. So due to the nature of constant
|
||
|
* item refreshing after each entity type is processed, it's safer to collect web items once more here where it
|
||
|
* is necessary. Collect host IDs and master item keys that cannot be resolved and then find web items and add
|
||
|
* references to item list.
|
||
|
*/
|
||
|
$unresolved_master_items = [];
|
||
|
$hostids = [];
|
||
|
|
||
|
foreach ($discovery_rules_by_hosts as $host => $discovery_rules) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($host);
|
||
|
$hostids[$hostid] = true;
|
||
|
|
||
|
foreach ($discovery_rules as $discovery_rule) {
|
||
|
if ($discovery_rule['type'] == ITEM_TYPE_DEPENDENT) {
|
||
|
if (!array_key_exists('key', $discovery_rule[$master_item_key])) {
|
||
|
throw new Exception( _s('Incorrect value for field "%1$s": %2$s.', 'master_itemid',
|
||
|
_('cannot be empty')
|
||
|
));
|
||
|
}
|
||
|
|
||
|
// if key cannot be resolved
|
||
|
if ($this->referencer->findItemidByKey($hostid,
|
||
|
$discovery_rule[$master_item_key]['key']) === null) {
|
||
|
$unresolved_master_items[$discovery_rule[$master_item_key]['key']] = true;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($unresolved_master_items) {
|
||
|
$items = API::Item()->get([
|
||
|
'output' => ['hostid', 'itemid', 'key_'],
|
||
|
'hostids' => array_keys($hostids),
|
||
|
'filter' => ['key_' => array_keys($unresolved_master_items)],
|
||
|
'webitems' => true
|
||
|
]);
|
||
|
|
||
|
foreach ($items as $item) {
|
||
|
$this->referencer->setDbItem($item['itemid'], $item);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($discovery_rules_by_hosts as $host => $discovery_rules) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($host);
|
||
|
|
||
|
foreach ($discovery_rules as $discovery_rule) {
|
||
|
$discovery_rule['hostid'] = $hostid;
|
||
|
|
||
|
$itemid = null;
|
||
|
|
||
|
if (array_key_exists('uuid', $discovery_rule)) {
|
||
|
$itemid = $this->referencer->findItemidByUuid($discovery_rule['uuid']);
|
||
|
}
|
||
|
|
||
|
if ($itemid === null) {
|
||
|
$itemid = $this->referencer->findItemidByKey($hostid, $discovery_rule['key_']);
|
||
|
}
|
||
|
|
||
|
unset($discovery_rule['item_prototypes'], $discovery_rule['trigger_prototypes'],
|
||
|
$discovery_rule['graph_prototypes'], $discovery_rule['host_prototypes']
|
||
|
);
|
||
|
|
||
|
$delay_types = [ITEM_TYPE_ZABBIX, ITEM_TYPE_SIMPLE, ITEM_TYPE_INTERNAL, ITEM_TYPE_ZABBIX_ACTIVE,
|
||
|
ITEM_TYPE_EXTERNAL, ITEM_TYPE_DB_MONITOR, ITEM_TYPE_IPMI, ITEM_TYPE_SSH, ITEM_TYPE_TELNET,
|
||
|
ITEM_TYPE_JMX, ITEM_TYPE_HTTPAGENT, ITEM_TYPE_SNMP, ITEM_TYPE_SCRIPT
|
||
|
];
|
||
|
|
||
|
if (!in_array($discovery_rule['type'], $delay_types)) {
|
||
|
unset($discovery_rule['delay']);
|
||
|
}
|
||
|
|
||
|
if (array_key_exists('interface_ref', $discovery_rule) && $discovery_rule['interface_ref']) {
|
||
|
$interfaceid = $this->referencer->findInterfaceidByRef($hostid, $discovery_rule['interface_ref']);
|
||
|
|
||
|
if ($interfaceid === null) {
|
||
|
throw new Exception(_s('Cannot find interface "%1$s" used for discovery rule "%2$s" on "%3$s".',
|
||
|
$discovery_rule['interface_ref'], $discovery_rule['name'], $host
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$discovery_rule['interfaceid'] = $interfaceid;
|
||
|
}
|
||
|
unset($discovery_rule['interface_ref']);
|
||
|
|
||
|
if ($discovery_rule['type'] == ITEM_TYPE_HTTPAGENT) {
|
||
|
$headers = [];
|
||
|
|
||
|
foreach ($discovery_rule['headers'] as $header) {
|
||
|
$headers[$header['name']] = $header['value'];
|
||
|
}
|
||
|
|
||
|
$discovery_rule['headers'] = $headers;
|
||
|
|
||
|
$query_fields = [];
|
||
|
|
||
|
foreach ($discovery_rule['query_fields'] as $query_field) {
|
||
|
$query_fields[] = [$query_field['name'] => $query_field['value']];
|
||
|
}
|
||
|
|
||
|
$discovery_rule['query_fields'] = $query_fields;
|
||
|
}
|
||
|
|
||
|
if ($discovery_rule['type'] == ITEM_TYPE_DEPENDENT) {
|
||
|
if (!array_key_exists('key', $discovery_rule[$master_item_key])) {
|
||
|
throw new Exception( _s('Incorrect value for field "%1$s": %2$s.', 'master_itemid',
|
||
|
_('cannot be empty')
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$discovery_rule['master_itemid'] = $this->referencer->findItemidByKey($hostid,
|
||
|
$discovery_rule[$master_item_key]['key'], true
|
||
|
);
|
||
|
}
|
||
|
|
||
|
unset($discovery_rule[$master_item_key]);
|
||
|
|
||
|
if ($discovery_rule['overrides']) {
|
||
|
foreach ($discovery_rule['overrides'] as &$override) {
|
||
|
foreach ($override['operations'] as &$operation) {
|
||
|
if ($operation['operationobject'] == OPERATION_OBJECT_HOST_PROTOTYPE
|
||
|
&& array_key_exists('optemplate', $operation)) {
|
||
|
foreach ($operation['optemplate'] as &$template) {
|
||
|
$templateid = $this->referencer->findTemplateidByHost($template['name']);
|
||
|
|
||
|
if ($templateid === null) {
|
||
|
throw new Exception(_s(
|
||
|
'Cannot find template "%1$s" for override "%2$s" of discovery rule "%3$s" on "%4$s".',
|
||
|
$template['name'],
|
||
|
$override['name'],
|
||
|
$discovery_rule['name'],
|
||
|
$host
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$template['templateid'] = $templateid;
|
||
|
unset($template['name']);
|
||
|
}
|
||
|
unset($template);
|
||
|
}
|
||
|
}
|
||
|
unset($operation);
|
||
|
}
|
||
|
unset($override);
|
||
|
}
|
||
|
|
||
|
foreach ($discovery_rule['preprocessing'] as &$preprocessing_step) {
|
||
|
$preprocessing_step['params'] = implode("\n", $preprocessing_step['parameters']);
|
||
|
|
||
|
unset($preprocessing_step['parameters']);
|
||
|
}
|
||
|
unset($preprocessing_step);
|
||
|
|
||
|
if (array_key_exists('filter', $discovery_rule)) {
|
||
|
foreach ($discovery_rule['filter']['conditions'] as &$condition) {
|
||
|
if ($discovery_rule['filter']['evaltype'] != CONDITION_EVAL_TYPE_EXPRESSION) {
|
||
|
unset($condition['formulaid']);
|
||
|
}
|
||
|
}
|
||
|
unset($condition);
|
||
|
}
|
||
|
|
||
|
foreach ($discovery_rule['overrides'] as &$override) {
|
||
|
if (!array_key_exists('filter', $override)) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
foreach ($override['filter']['conditions'] as &$condition) {
|
||
|
if ($override['filter']['evaltype'] != CONDITION_EVAL_TYPE_EXPRESSION) {
|
||
|
unset($condition['formulaid']);
|
||
|
}
|
||
|
}
|
||
|
unset($condition);
|
||
|
}
|
||
|
unset($override);
|
||
|
|
||
|
if ($itemid !== null) {
|
||
|
$discovery_rule['itemid'] = $itemid;
|
||
|
$discovery_rules_to_update[] = array_diff_key($discovery_rule, array_flip(['hostid']));
|
||
|
$upd_discovery_rule_hostids[] = $discovery_rule['hostid'];
|
||
|
}
|
||
|
else {
|
||
|
/*
|
||
|
* The array key "lld_macro_paths" must exist at this point. It is processed by chain conversion.
|
||
|
* Unlike discoveryrule.update method, discoveryrule.create does not allow "lld_macro_paths"
|
||
|
* to be empty.
|
||
|
*/
|
||
|
if (!$discovery_rule['lld_macro_paths']) {
|
||
|
unset($discovery_rule['lld_macro_paths']);
|
||
|
}
|
||
|
$discovery_rules_to_create[] = $discovery_rule;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$processed_discovery_rules = [];
|
||
|
|
||
|
if ($this->options['discoveryRules']['createMissing'] && $discovery_rules_to_create) {
|
||
|
API::DiscoveryRule()->create($discovery_rules_to_create);
|
||
|
|
||
|
foreach ($discovery_rules_to_create as $discovery_rule) {
|
||
|
$processed_discovery_rules[$discovery_rule['hostid']][$discovery_rule['key_']] = 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($this->options['discoveryRules']['updateExisting'] && $discovery_rules_to_update) {
|
||
|
API::DiscoveryRule()->update($discovery_rules_to_update);
|
||
|
|
||
|
foreach ($discovery_rules_to_update as $discovery_rule) {
|
||
|
$hostid = array_shift($upd_discovery_rule_hostids);
|
||
|
|
||
|
$processed_discovery_rules[$hostid][$discovery_rule['key_']] = 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Refresh discovery rules because templated ones can be inherited to host and used for prototypes.
|
||
|
$this->referencer->refreshItems();
|
||
|
|
||
|
$order_tree = $this->getDiscoveryRulesItemsOrder($master_item_key);
|
||
|
|
||
|
// process prototypes
|
||
|
$item_prototypes_to_update = [];
|
||
|
$item_prototypes_to_create = [];
|
||
|
$host_prototypes_to_update = [];
|
||
|
$host_prototypes_to_create = [];
|
||
|
$levels = [];
|
||
|
|
||
|
foreach ($discovery_rules_by_hosts as $host => $discovery_rules) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($host);
|
||
|
|
||
|
foreach ($discovery_rules as $discovery_rule) {
|
||
|
// if rule was not processed we should not create/update any of its prototypes
|
||
|
if (!array_key_exists($hostid, $processed_discovery_rules)
|
||
|
|| !array_key_exists($discovery_rule['key_'], $processed_discovery_rules[$hostid])) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$itemid = $this->referencer->findItemidByKey($hostid, $discovery_rule['key_']);
|
||
|
|
||
|
// prototypes
|
||
|
$item_prototypes = $discovery_rule['item_prototypes'] ? $order_tree[$host][$discovery_rule['key_']] : [];
|
||
|
|
||
|
foreach ($item_prototypes as $index => $level) {
|
||
|
$item_prototype = $discovery_rule['item_prototypes'][$index];
|
||
|
$item_prototype['hostid'] = $hostid;
|
||
|
unset($item_prototype['trigger_prototypes']);
|
||
|
|
||
|
$levels[$level] = true;
|
||
|
|
||
|
$delay_types = [ITEM_TYPE_ZABBIX, ITEM_TYPE_SIMPLE, ITEM_TYPE_INTERNAL, ITEM_TYPE_ZABBIX_ACTIVE,
|
||
|
ITEM_TYPE_EXTERNAL, ITEM_TYPE_DB_MONITOR, ITEM_TYPE_IPMI, ITEM_TYPE_SSH, ITEM_TYPE_TELNET,
|
||
|
ITEM_TYPE_CALCULATED, ITEM_TYPE_JMX, ITEM_TYPE_HTTPAGENT, ITEM_TYPE_SNMP, ITEM_TYPE_SCRIPT
|
||
|
];
|
||
|
|
||
|
if (!in_array($item_prototype['type'], $delay_types)) {
|
||
|
unset($item_prototype['delay']);
|
||
|
}
|
||
|
|
||
|
if (array_key_exists('interface_ref', $item_prototype) && $item_prototype['interface_ref']) {
|
||
|
$interfaceid = $this->referencer->findInterfaceidByRef($hostid,
|
||
|
$item_prototype['interface_ref']
|
||
|
);
|
||
|
|
||
|
if ($interfaceid !== null) {
|
||
|
$item_prototype['interfaceid'] = $interfaceid;
|
||
|
}
|
||
|
else {
|
||
|
throw new Exception(_s(
|
||
|
'Cannot find interface "%1$s" used for item prototype "%2$s" of discovery rule "%3$s" on "%4$s".',
|
||
|
$item_prototype['interface_ref'],
|
||
|
$item_prototype['name'],
|
||
|
$discovery_rule['name'],
|
||
|
$host
|
||
|
));
|
||
|
}
|
||
|
}
|
||
|
unset($item_prototype['interface_ref']);
|
||
|
|
||
|
if ($item_prototype['valuemap']) {
|
||
|
$valuemapid = $this->referencer->findValuemapidByName($hostid,
|
||
|
$item_prototype['valuemap']['name']
|
||
|
);
|
||
|
|
||
|
if ($valuemapid === null) {
|
||
|
throw new Exception(_s(
|
||
|
'Cannot find value map "%1$s" used for item prototype "%2$s" of discovery rule "%3$s" on "%4$s".',
|
||
|
$item_prototype['valuemap']['name'],
|
||
|
$item_prototype['name'],
|
||
|
$discovery_rule['name'],
|
||
|
$host
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$item_prototype['valuemapid'] = $valuemapid;
|
||
|
}
|
||
|
unset($item_prototype['valuemap']);
|
||
|
|
||
|
if ($item_prototype['type'] == ITEM_TYPE_DEPENDENT) {
|
||
|
if (!array_key_exists('key', $item_prototype[$master_item_key])) {
|
||
|
throw new Exception( _s('Incorrect value for field "%1$s": %2$s.', 'master_itemid',
|
||
|
_('cannot be empty')
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$master_item_prototypeid = $this->referencer->findItemidByKey($hostid,
|
||
|
$item_prototype[$master_item_key]['key'], true
|
||
|
);
|
||
|
|
||
|
if ($master_item_prototypeid !== null) {
|
||
|
$item_prototype['master_itemid'] = $master_item_prototypeid;
|
||
|
unset($item_prototype[$master_item_key]);
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
unset($item_prototype[$master_item_key]);
|
||
|
}
|
||
|
|
||
|
if ($item_prototype['type'] == ITEM_TYPE_HTTPAGENT) {
|
||
|
$headers = [];
|
||
|
|
||
|
foreach ($item_prototype['headers'] as $header) {
|
||
|
$headers[$header['name']] = $header['value'];
|
||
|
}
|
||
|
|
||
|
$item_prototype['headers'] = $headers;
|
||
|
|
||
|
$query_fields = [];
|
||
|
|
||
|
foreach ($item_prototype['query_fields'] as $query_field) {
|
||
|
$query_fields[] = [$query_field['name'] => $query_field['value']];
|
||
|
}
|
||
|
|
||
|
$item_prototype['query_fields'] = $query_fields;
|
||
|
}
|
||
|
|
||
|
$item_prototypeid = null;
|
||
|
|
||
|
if (array_key_exists('uuid', $item_prototype)) {
|
||
|
$item_prototypeid = $this->referencer->findItemidByUuid($item_prototype['uuid']);
|
||
|
}
|
||
|
|
||
|
if ($item_prototypeid === null) {
|
||
|
$item_prototypeid = $this->referencer->findItemidByKey($hostid, $item_prototype['key_']);
|
||
|
}
|
||
|
|
||
|
if ($item_prototypeid === null) {
|
||
|
$item_prototype['ruleid'] = $itemid;
|
||
|
}
|
||
|
|
||
|
foreach ($item_prototype['preprocessing'] as &$preprocessing_step) {
|
||
|
$preprocessing_step['params'] = implode("\n", $preprocessing_step['parameters']);
|
||
|
|
||
|
unset($preprocessing_step['parameters']);
|
||
|
}
|
||
|
unset($preprocessing_step);
|
||
|
|
||
|
if ($item_prototypeid !== null) {
|
||
|
if (!array_key_exists($level, $item_prototypes_to_update)) {
|
||
|
$item_prototypes_to_update[$level] = [];
|
||
|
}
|
||
|
$item_prototype['itemid'] = $item_prototypeid;
|
||
|
$item_prototypes_to_update[$level][] = $item_prototype;
|
||
|
}
|
||
|
else {
|
||
|
if (!array_key_exists($level, $item_prototypes_to_create)) {
|
||
|
$item_prototypes_to_create[$level] = [];
|
||
|
}
|
||
|
$item_prototypes_to_create[$level][] = $item_prototype;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($discovery_rule['host_prototypes'] as $host_prototype) {
|
||
|
// Resolve group prototypes.
|
||
|
$group_links = [];
|
||
|
|
||
|
foreach ($host_prototype['group_links'] as $group_link) {
|
||
|
$groupid = $this->referencer->findHostGroupidByName($group_link['group']['name']);
|
||
|
|
||
|
if ($groupid === null) {
|
||
|
throw new Exception(_s(
|
||
|
'Cannot find host group "%1$s" for host prototype "%2$s" of discovery rule "%3$s" on "%4$s".',
|
||
|
$group_link['group']['name'],
|
||
|
$host_prototype['name'],
|
||
|
$discovery_rule['name'],
|
||
|
$host
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$group_links[] = ['groupid' => $groupid];
|
||
|
}
|
||
|
|
||
|
$host_prototype['groupLinks'] = $group_links;
|
||
|
$host_prototype['groupPrototypes'] = $host_prototype['group_prototypes'];
|
||
|
unset($host_prototype['group_links'], $host_prototype['group_prototypes']);
|
||
|
|
||
|
// Resolve templates.
|
||
|
$templates = [];
|
||
|
|
||
|
foreach ($host_prototype['templates'] as $template) {
|
||
|
$templateid = $this->referencer->findTemplateidByHost($template['name']);
|
||
|
|
||
|
if ($templateid === null) {
|
||
|
throw new Exception(_s(
|
||
|
'Cannot find template "%1$s" for host prototype "%2$s" of discovery rule "%3$s" on "%4$s".',
|
||
|
$template['name'],
|
||
|
$host_prototype['name'],
|
||
|
$discovery_rule['name'],
|
||
|
$host
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$templates[] = ['templateid' => $templateid];
|
||
|
}
|
||
|
|
||
|
$host_prototype['templates'] = $templates;
|
||
|
|
||
|
$host_prototypeid = null;
|
||
|
|
||
|
if (array_key_exists('uuid', $host_prototype)) {
|
||
|
$host_prototypeid = $this->referencer->findHostPrototypeidByUuid($host_prototype['uuid']);
|
||
|
}
|
||
|
|
||
|
if ($host_prototypeid === null) {
|
||
|
$host_prototypeid = $this->referencer->findHostPrototypeidByHost($hostid, $itemid,
|
||
|
$host_prototype['host']
|
||
|
);
|
||
|
}
|
||
|
|
||
|
if ($host_prototypeid !== null) {
|
||
|
if (array_key_exists('macros', $host_prototype)) {
|
||
|
foreach ($host_prototype['macros'] as &$macro) {
|
||
|
$hostmacroid = $this->referencer->findHostPrototypeMacroid($host_prototypeid,
|
||
|
$macro['macro']
|
||
|
);
|
||
|
|
||
|
if ($hostmacroid !== null) {
|
||
|
$macro['hostmacroid'] = $hostmacroid;
|
||
|
}
|
||
|
}
|
||
|
unset($macro);
|
||
|
}
|
||
|
|
||
|
$host_prototype['hostid'] = $host_prototypeid;
|
||
|
$host_prototypes_to_update[] = $host_prototype;
|
||
|
}
|
||
|
else {
|
||
|
$host_prototype['ruleid'] = $itemid;
|
||
|
$host_prototypes_to_create[] = $host_prototype;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ksort($levels);
|
||
|
foreach (array_keys($levels) as $level) {
|
||
|
if (array_key_exists($level, $item_prototypes_to_update) && $item_prototypes_to_update[$level]) {
|
||
|
$this->updateItemsWithDependency([$item_prototypes_to_update[$level]], $master_item_key,
|
||
|
API::ItemPrototype()
|
||
|
);
|
||
|
}
|
||
|
if (array_key_exists($level, $item_prototypes_to_create) && $item_prototypes_to_create[$level]) {
|
||
|
$this->createItemsWithDependency([$item_prototypes_to_create[$level]], $master_item_key,
|
||
|
API::ItemPrototype()
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($host_prototypes_to_update) {
|
||
|
API::HostPrototype()->update($host_prototypes_to_update);
|
||
|
}
|
||
|
|
||
|
if ($host_prototypes_to_create) {
|
||
|
API::HostPrototype()->create($host_prototypes_to_create);
|
||
|
}
|
||
|
|
||
|
// Refresh prototypes because templated ones can be inherited to host and used in triggers prototypes or graph
|
||
|
// prototypes.
|
||
|
$this->referencer->refreshItems();
|
||
|
|
||
|
// First we need to create item prototypes and only then trigger and graph prototypes.
|
||
|
$triggers_to_create = [];
|
||
|
$triggers_to_update = [];
|
||
|
$graphs_to_create = [];
|
||
|
$graphs_to_update = [];
|
||
|
|
||
|
// The list of triggers to process dependencies.
|
||
|
$triggers = [];
|
||
|
|
||
|
foreach ($discovery_rules_by_hosts as $host => $discovery_rules) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($host);
|
||
|
|
||
|
foreach ($discovery_rules as $discovery_rule) {
|
||
|
// If rule was not processed we should not create/update any of its prototypes.
|
||
|
if (array_key_exists($hostid, $processed_discovery_rules)
|
||
|
&& !array_key_exists($discovery_rule['key_'], $processed_discovery_rules[$hostid])) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
foreach ($discovery_rule['trigger_prototypes'] as $trigger) {
|
||
|
$triggerid = null;
|
||
|
|
||
|
if (array_key_exists('uuid', $trigger)) {
|
||
|
$triggerid = $this->referencer->findTriggeridByUuid($trigger['uuid']);
|
||
|
}
|
||
|
|
||
|
if ($triggerid === null) {
|
||
|
$triggerid = $this->referencer->findTriggeridByName($trigger['description'],
|
||
|
$trigger['expression'], $trigger['recovery_expression']
|
||
|
);
|
||
|
}
|
||
|
|
||
|
$triggers[] = $trigger;
|
||
|
unset($trigger['dependencies']);
|
||
|
|
||
|
if ($triggerid !== null) {
|
||
|
$trigger['triggerid'] = $triggerid;
|
||
|
$triggers_to_update[] = $trigger;
|
||
|
}
|
||
|
else {
|
||
|
$triggers_to_create[] = $trigger;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($discovery_rule['graph_prototypes'] as $graph) {
|
||
|
if ($graph['ymin_item_1']) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($graph['ymin_item_1']['host']);
|
||
|
|
||
|
$itemid = ($hostid !== null)
|
||
|
? $this->referencer->findItemidByKey($hostid, $graph['ymin_item_1']['key'], true)
|
||
|
: null;
|
||
|
|
||
|
if ($itemid === null) {
|
||
|
throw new Exception(_s(
|
||
|
'Cannot find item "%1$s" on "%2$s" used as the Y axis MIN value for graph prototype "%3$s" of discovery rule "%4$s" on "%5$s".',
|
||
|
$graph['ymin_item_1']['key'],
|
||
|
$graph['ymin_item_1']['host'],
|
||
|
$graph['name'],
|
||
|
$discovery_rule['name'],
|
||
|
$host
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$graph['ymin_itemid'] = $itemid;
|
||
|
}
|
||
|
|
||
|
if ($graph['ymax_item_1']) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($graph['ymax_item_1']['host']);
|
||
|
|
||
|
$itemid = ($hostid !== null)
|
||
|
? $this->referencer->findItemidByKey($hostid, $graph['ymax_item_1']['key'], true)
|
||
|
: null;
|
||
|
|
||
|
if ($itemid === null) {
|
||
|
throw new Exception(_s(
|
||
|
'Cannot find item "%1$s" on "%2$s" used as the Y axis MAX value for graph prototype "%3$s" of discovery rule "%4$s" on "%5$s".',
|
||
|
$graph['ymax_item_1']['key'],
|
||
|
$graph['ymax_item_1']['host'],
|
||
|
$graph['name'],
|
||
|
$discovery_rule['name'],
|
||
|
$host
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$graph['ymax_itemid'] = $itemid;
|
||
|
}
|
||
|
|
||
|
foreach ($graph['gitems'] as &$item) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($item['item']['host']);
|
||
|
|
||
|
$item['itemid'] = ($hostid !== null)
|
||
|
? $this->referencer->findItemidByKey($hostid, $item['item']['key'], true)
|
||
|
: null;
|
||
|
|
||
|
if ($item['itemid'] === null) {
|
||
|
throw new Exception(_s(
|
||
|
'Cannot find item "%1$s" on "%2$s" used in graph prototype "%3$s" of discovery rule "%4$s" on "%5$s".',
|
||
|
$item['item']['key'],
|
||
|
$item['item']['host'],
|
||
|
$graph['name'],
|
||
|
$discovery_rule['name'],
|
||
|
$host
|
||
|
));
|
||
|
}
|
||
|
}
|
||
|
unset($item);
|
||
|
|
||
|
$graphid = null;
|
||
|
|
||
|
if (array_key_exists('uuid', $graph)) {
|
||
|
$graphid = $this->referencer->findGraphidByUuid($graph['uuid']);
|
||
|
}
|
||
|
|
||
|
if ($graphid === null) {
|
||
|
$graphid = $this->referencer->findGraphidByName($hostid, $graph['name']);
|
||
|
}
|
||
|
|
||
|
if ($graphid !== null) {
|
||
|
$graph['graphid'] = $graphid;
|
||
|
$graphs_to_update[] = $graph;
|
||
|
}
|
||
|
else {
|
||
|
$graphs_to_create[] = $graph;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($triggers_to_update) {
|
||
|
$updated_triggers = API::TriggerPrototype()->update($triggers_to_update);
|
||
|
|
||
|
foreach ($updated_triggers['triggerids'] as $index => $triggerid) {
|
||
|
$trigger = $triggers_to_update[$index];
|
||
|
$this->referencer->setDbTrigger($triggerid, $trigger);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($triggers_to_create) {
|
||
|
$created_triggers = API::TriggerPrototype()->create($triggers_to_create);
|
||
|
|
||
|
foreach ($created_triggers['triggerids'] as $index => $triggerid) {
|
||
|
$trigger = $triggers_to_create[$index];
|
||
|
$this->referencer->setDbTrigger($triggerid, $trigger);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($graphs_to_update) {
|
||
|
API::GraphPrototype()->update($graphs_to_update);
|
||
|
$this->referencer->refreshGraphs();
|
||
|
}
|
||
|
|
||
|
if ($graphs_to_create) {
|
||
|
API::GraphPrototype()->create($graphs_to_create);
|
||
|
$this->referencer->refreshGraphs();
|
||
|
}
|
||
|
|
||
|
$this->processTriggerPrototypeDependencies($triggers);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Update trigger prototype dependencies.
|
||
|
*
|
||
|
* @param array $triggers
|
||
|
*
|
||
|
* @throws Exception
|
||
|
*/
|
||
|
protected function processTriggerPrototypeDependencies(array $triggers): void {
|
||
|
$trigger_dependencies = [];
|
||
|
|
||
|
foreach ($triggers as $trigger) {
|
||
|
if (!array_key_exists('dependencies', $trigger)) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$dependencies = [];
|
||
|
$triggerid = $this->referencer->findTriggeridByName($trigger['description'], $trigger['expression'],
|
||
|
$trigger['recovery_expression']
|
||
|
);
|
||
|
|
||
|
foreach ($trigger['dependencies'] as $dependency) {
|
||
|
$dependent_triggerid = $this->referencer->findTriggeridByName($dependency['name'],
|
||
|
$dependency['expression'], $dependency['recovery_expression'], true
|
||
|
);
|
||
|
|
||
|
if ($dependent_triggerid === null) {
|
||
|
throw new Exception(_s('Trigger prototype "%1$s" depends on trigger "%2$s", which does not exist.',
|
||
|
$trigger['description'],
|
||
|
$dependency['name']
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$dependencies[] = ['triggerid' => $dependent_triggerid];
|
||
|
}
|
||
|
|
||
|
$trigger_dependencies[] = [
|
||
|
'triggerid' => $triggerid,
|
||
|
'dependencies' => $dependencies
|
||
|
];
|
||
|
}
|
||
|
|
||
|
if ($trigger_dependencies) {
|
||
|
API::TriggerPrototype()->update($trigger_dependencies);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Import web scenarios.
|
||
|
*
|
||
|
* @throws APIException
|
||
|
*/
|
||
|
protected function processHttpTests(): void {
|
||
|
if (!$this->options['httptests']['createMissing'] && !$this->options['httptests']['updateExisting']) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$httptests_to_create = [];
|
||
|
$httptests_to_update = [];
|
||
|
|
||
|
foreach ($this->getFormattedHttpTests() as $host => $httptests) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($host);
|
||
|
|
||
|
if ($hostid === null
|
||
|
|| (!$this->importedObjectContainer->isHostProcessed($hostid)
|
||
|
&& !$this->importedObjectContainer->isTemplateProcessed($hostid))) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
foreach ($httptests as $httptest) {
|
||
|
$httptestid = null;
|
||
|
|
||
|
if (array_key_exists('uuid', $httptest)) {
|
||
|
$httptestid = $this->referencer->findHttpTestidByUuid($httptest['uuid']);
|
||
|
}
|
||
|
|
||
|
if ($httptestid === null) {
|
||
|
$httptestid = $this->referencer->findHttpTestidByName($hostid, $httptest['name']);
|
||
|
}
|
||
|
|
||
|
if ($httptestid !== null) {
|
||
|
foreach ($httptest['steps'] as &$httpstep) {
|
||
|
$httpstepid = $this->referencer->findHttpStepidByName($hostid, $httptestid, $httpstep['name']);
|
||
|
|
||
|
if ($httpstepid !== null) {
|
||
|
$httpstep['httpstepid'] = $httpstepid;
|
||
|
}
|
||
|
}
|
||
|
unset($httpstep);
|
||
|
|
||
|
$httptest['httptestid'] = $httptestid;
|
||
|
$httptests_to_update[] = $httptest;
|
||
|
}
|
||
|
else {
|
||
|
$httptest['hostid'] = $hostid;
|
||
|
$httptests_to_create[] = $httptest;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($this->options['httptests']['updateExisting'] && $httptests_to_update) {
|
||
|
API::HttpTest()->update($httptests_to_update);
|
||
|
}
|
||
|
|
||
|
if ($this->options['httptests']['createMissing'] && $httptests_to_create) {
|
||
|
API::HttpTest()->create($httptests_to_create);
|
||
|
}
|
||
|
|
||
|
$this->referencer->refreshHttpTests();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Import graphs.
|
||
|
*
|
||
|
* @throws Exception
|
||
|
*/
|
||
|
protected function processGraphs(): void {
|
||
|
if (!$this->options['graphs']['createMissing'] && !$this->options['graphs']['updateExisting']) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$graphs_to_create = [];
|
||
|
$graphs_to_update = [];
|
||
|
|
||
|
foreach ($this->getFormattedGraphs() as $graph) {
|
||
|
if ($graph['ymin_item_1']) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($graph['ymin_item_1']['host']);
|
||
|
$itemid = ($hostid !== null)
|
||
|
? $this->referencer->findItemidByKey($hostid, $graph['ymin_item_1']['key'], true)
|
||
|
: null;
|
||
|
|
||
|
if ($itemid === null) {
|
||
|
throw new Exception(_s(
|
||
|
'Cannot find item "%1$s" on "%2$s" used as the Y axis MIN value for graph "%3$s".',
|
||
|
$graph['ymin_item_1']['key'],
|
||
|
$graph['ymin_item_1']['host'],
|
||
|
$graph['name']
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$graph['ymin_itemid'] = $itemid;
|
||
|
}
|
||
|
|
||
|
if ($graph['ymax_item_1']) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($graph['ymax_item_1']['host']);
|
||
|
$itemid = ($hostid !== null)
|
||
|
? $this->referencer->findItemidByKey($hostid, $graph['ymax_item_1']['key'], true)
|
||
|
: null;
|
||
|
|
||
|
if ($itemid === null) {
|
||
|
throw new Exception(_s(
|
||
|
'Cannot find item "%1$s" on "%2$s" used as the Y axis MAX value for graph "%3$s".',
|
||
|
$graph['ymax_item_1']['key'],
|
||
|
$graph['ymax_item_1']['host'],
|
||
|
$graph['name']
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$graph['ymax_itemid'] = $itemid;
|
||
|
}
|
||
|
|
||
|
$hostid = null;
|
||
|
|
||
|
foreach ($graph['gitems'] as &$item) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($item['item']['host']);
|
||
|
$item['itemid'] = ($hostid !== null)
|
||
|
? $this->referencer->findItemidByKey($hostid, $item['item']['key'], true)
|
||
|
: null;
|
||
|
|
||
|
if ($item['itemid'] === null) {
|
||
|
throw new Exception(_s(
|
||
|
'Cannot find item "%1$s" on "%2$s" used in graph "%3$s".',
|
||
|
$item['item']['key'],
|
||
|
$item['item']['host'],
|
||
|
$graph['name']
|
||
|
));
|
||
|
}
|
||
|
}
|
||
|
unset($item);
|
||
|
|
||
|
if ($this->isTemplateGraph($graph)) {
|
||
|
$graphid = $this->referencer->findGraphidByUuid($graph['uuid']);
|
||
|
|
||
|
if ($graphid === null) {
|
||
|
$graphid = $this->referencer->findGraphidByName($hostid, $graph['name']);
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
unset($graph['uuid']);
|
||
|
$graphid = $this->referencer->findGraphidByName($hostid, $graph['name']);
|
||
|
}
|
||
|
|
||
|
if ($graphid !== null) {
|
||
|
$graph['graphid'] = $graphid;
|
||
|
$graphs_to_update[] = $graph;
|
||
|
}
|
||
|
else {
|
||
|
$graphs_to_create[] = $graph;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($this->options['graphs']['updateExisting'] && $graphs_to_update) {
|
||
|
API::Graph()->update($graphs_to_update);
|
||
|
}
|
||
|
|
||
|
if ($this->options['graphs']['createMissing'] && $graphs_to_create) {
|
||
|
API::Graph()->create($graphs_to_create);
|
||
|
}
|
||
|
|
||
|
$this->referencer->refreshGraphs();
|
||
|
}
|
||
|
|
||
|
private function isTemplateGraph(array $graph): bool {
|
||
|
if ($graph['ymin_item_1'] && $this->referencer->findTemplateidByHost($graph['ymin_item_1']['host'])) {
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
if ($graph['ymax_item_1'] && $this->referencer->findTemplateidByHost($graph['ymax_item_1']['host'])) {
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
if (array_key_exists('gitems', $graph) && $graph['gitems']) {
|
||
|
foreach ($graph['gitems'] as $gitem) {
|
||
|
if ($this->referencer->findTemplateidByHost($gitem['item']['host'])) {
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Import triggers.
|
||
|
*
|
||
|
* @throws Exception
|
||
|
*/
|
||
|
protected function processTriggers(): void {
|
||
|
if (!$this->options['triggers']['createMissing'] && !$this->options['triggers']['updateExisting']) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$triggers_to_create = [];
|
||
|
$triggers_to_update = [];
|
||
|
|
||
|
$triggers_to_process_dependencies = [];
|
||
|
|
||
|
foreach ($this->getFormattedTriggers() as $trigger) {
|
||
|
$triggerid = null;
|
||
|
|
||
|
$is_template_trigger = $this->isTemplateTrigger($trigger);
|
||
|
|
||
|
if ($is_template_trigger && array_key_exists('uuid', $trigger)) {
|
||
|
$triggerid = $this->referencer->findTriggeridByUuid($trigger['uuid']);
|
||
|
|
||
|
if ($triggerid === null) {
|
||
|
$triggerid = $this->referencer->findTriggeridByName($trigger['description'], $trigger['expression'],
|
||
|
$trigger['recovery_expression']
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
elseif (!$is_template_trigger) {
|
||
|
unset($trigger['uuid']);
|
||
|
$triggerid = $this->referencer->findTriggeridByName($trigger['description'], $trigger['expression'],
|
||
|
$trigger['recovery_expression']
|
||
|
);
|
||
|
}
|
||
|
|
||
|
if ($triggerid !== null) {
|
||
|
if ($this->options['triggers']['updateExisting']) {
|
||
|
$triggers_to_process_dependencies[] = $trigger;
|
||
|
|
||
|
$trigger['triggerid'] = $triggerid;
|
||
|
unset($trigger['dependencies']);
|
||
|
$triggers_to_update[] = $trigger;
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
if ($this->options['triggers']['createMissing']) {
|
||
|
$triggers_to_process_dependencies[] = $trigger;
|
||
|
|
||
|
unset($trigger['dependencies']);
|
||
|
$triggers_to_create[] = $trigger;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($triggers_to_update) {
|
||
|
API::Trigger()->update($triggers_to_update);
|
||
|
}
|
||
|
|
||
|
if ($triggers_to_create) {
|
||
|
API::Trigger()->create($triggers_to_create);
|
||
|
}
|
||
|
|
||
|
// Refresh triggers because template triggers can be inherited to host and used in maps.
|
||
|
$this->referencer->refreshTriggers();
|
||
|
|
||
|
$this->processTriggerDependencies($triggers_to_process_dependencies);
|
||
|
}
|
||
|
|
||
|
private function isTemplateTrigger(array $trigger): bool {
|
||
|
$expression_parser = new CExpressionParser(['usermacros' => true]);
|
||
|
|
||
|
if ($expression_parser->parse($trigger['expression']) != CParser::PARSE_SUCCESS) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
foreach ($expression_parser->getResult()->getHosts() as $host) {
|
||
|
$host = $this->referencer->findTemplateidByHost($host);
|
||
|
|
||
|
if ($host !== null) {
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($trigger['recovery_expression'] === ''
|
||
|
|| $expression_parser->parse($trigger['recovery_expression']) != CParser::PARSE_SUCCESS) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
foreach ($expression_parser->getResult()->getHosts() as $host) {
|
||
|
$host = $this->referencer->findTemplateidByHost($host);
|
||
|
|
||
|
if ($host !== null) {
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Update trigger dependencies
|
||
|
*
|
||
|
* @param array $triggers
|
||
|
*
|
||
|
* @throws Exception
|
||
|
*/
|
||
|
protected function processTriggerDependencies(array $triggers): void {
|
||
|
$trigger_dependencies = [];
|
||
|
|
||
|
foreach ($triggers as $trigger) {
|
||
|
if (!array_key_exists('dependencies', $trigger)) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$triggerid = $this->referencer->findTriggeridByName($trigger['description'], $trigger['expression'],
|
||
|
$trigger['recovery_expression']
|
||
|
);
|
||
|
|
||
|
$dependencies = [];
|
||
|
|
||
|
foreach ($trigger['dependencies'] as $dependency) {
|
||
|
$dependent_triggerid = $this->referencer->findTriggeridByName($dependency['name'],
|
||
|
$dependency['expression'], $dependency['recovery_expression'], true
|
||
|
);
|
||
|
|
||
|
if ($dependent_triggerid === null) {
|
||
|
throw new Exception(_s('Trigger "%1$s" depends on trigger "%2$s", which does not exist.',
|
||
|
$trigger['description'],
|
||
|
$dependency['name']
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$dependencies[] = ['triggerid' => $dependent_triggerid];
|
||
|
}
|
||
|
|
||
|
$trigger_dependencies[] = [
|
||
|
'triggerid' => $triggerid,
|
||
|
'dependencies' => $dependencies
|
||
|
];
|
||
|
}
|
||
|
|
||
|
if ($trigger_dependencies) {
|
||
|
API::Trigger()->update($trigger_dependencies);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Import images.
|
||
|
*
|
||
|
* @throws Exception
|
||
|
*/
|
||
|
protected function processImages(): void {
|
||
|
if (!$this->options['images']['updateExisting'] && !$this->options['images']['createMissing']) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$images_to_import = $this->getFormattedImages();
|
||
|
|
||
|
if (!$images_to_import) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$images_to_update = [];
|
||
|
$images_to_create = [];
|
||
|
|
||
|
foreach ($images_to_import as $image) {
|
||
|
$imageid = $this->referencer->findImageidByName($image['name']);
|
||
|
|
||
|
if ($imageid !== null) {
|
||
|
$image['imageid'] = $imageid;
|
||
|
unset($image['imagetype']);
|
||
|
$images_to_update[] = $image;
|
||
|
}
|
||
|
else {
|
||
|
$images_to_create[] = $image;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($this->options['images']['updateExisting'] && $images_to_update) {
|
||
|
API::Image()->update($images_to_update);
|
||
|
}
|
||
|
|
||
|
if ($this->options['images']['createMissing'] && $images_to_create) {
|
||
|
$created_images = API::Image()->create($images_to_create);
|
||
|
|
||
|
foreach ($images_to_create as $index => $image) {
|
||
|
$this->referencer->setDbImage($created_images['imageids'][$index], $image);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Import maps.
|
||
|
*
|
||
|
* @throws Exception
|
||
|
*/
|
||
|
protected function processMaps(): void {
|
||
|
if ($this->options['maps']['updateExisting'] || $this->options['maps']['createMissing']) {
|
||
|
$maps = $this->getFormattedMaps();
|
||
|
|
||
|
if ($maps) {
|
||
|
$map_importer = new CMapImporter($this->options, $this->referencer, $this->importedObjectContainer);
|
||
|
$map_importer->import($maps);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Import template dashboards.
|
||
|
*/
|
||
|
protected function processTemplateDashboards(): void {
|
||
|
if ($this->options['templateDashboards']['updateExisting']
|
||
|
|| $this->options['templateDashboards']['createMissing']) {
|
||
|
$dashboards = $this->getFormattedTemplateDashboards();
|
||
|
|
||
|
if ($dashboards) {
|
||
|
$dashboard_importer = new CTemplateDashboardImporter($this->options, $this->referencer,
|
||
|
$this->importedObjectContainer
|
||
|
);
|
||
|
|
||
|
$dashboard_importer->import($dashboards);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Import media types.
|
||
|
*/
|
||
|
protected function processMediaTypes(): void {
|
||
|
if (!$this->options['mediaTypes']['updateExisting'] && !$this->options['mediaTypes']['createMissing']) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$media_types_to_import = $this->getFormattedMediaTypes();
|
||
|
|
||
|
if (!$media_types_to_import) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$media_types_to_import = zbx_toHash($media_types_to_import, 'name');
|
||
|
|
||
|
$db_media_types = API::MediaType()->get([
|
||
|
'output' => ['mediatypeid', 'name'],
|
||
|
'filter' => ['name' => array_keys($media_types_to_import)]
|
||
|
]);
|
||
|
$db_media_types = zbx_toHash($db_media_types, 'name');
|
||
|
|
||
|
$media_types_to_update = [];
|
||
|
$media_types_to_create = [];
|
||
|
|
||
|
foreach ($media_types_to_import as $name => $media_type) {
|
||
|
if (array_key_exists($name, $db_media_types)) {
|
||
|
$media_type['mediatypeid'] = $db_media_types[$name]['mediatypeid'];
|
||
|
$media_types_to_update[] = $media_type;
|
||
|
}
|
||
|
else {
|
||
|
$media_types_to_create[] = $media_type;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($this->options['mediaTypes']['updateExisting'] && $media_types_to_update) {
|
||
|
API::MediaType()->update($media_types_to_update);
|
||
|
}
|
||
|
|
||
|
if ($this->options['mediaTypes']['createMissing'] && $media_types_to_create) {
|
||
|
API::MediaType()->create($media_types_to_create);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Deletes items from DB that are missing in import file.
|
||
|
*/
|
||
|
protected function deleteMissingItems(): void {
|
||
|
if (!$this->options['items']['deleteMissing']) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$processed_hostids = array_merge(
|
||
|
$this->importedObjectContainer->getHostids(),
|
||
|
$this->importedObjectContainer->getTemplateids()
|
||
|
);
|
||
|
|
||
|
if (!$processed_hostids) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$itemids = [];
|
||
|
|
||
|
foreach ($this->getFormattedItems() as $host => $items) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($host);
|
||
|
|
||
|
if ($hostid === null) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
foreach ($items as $item) {
|
||
|
$itemid = null;
|
||
|
|
||
|
if (array_key_exists('uuid', $item)) {
|
||
|
$itemid = $this->referencer->findItemidByUuid($item['uuid']);
|
||
|
}
|
||
|
|
||
|
if ($itemid === null) {
|
||
|
$itemid = $this->referencer->findItemidByKey($hostid, $item['key_']);
|
||
|
}
|
||
|
|
||
|
if ($itemid !== null) {
|
||
|
$itemids[$itemid] = [];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$db_itemids = API::Item()->get([
|
||
|
'output' => ['itemid'],
|
||
|
'hostids' => $processed_hostids,
|
||
|
'filter' => ['flags' => ZBX_FLAG_DISCOVERY_NORMAL],
|
||
|
'inherited' => false,
|
||
|
'preservekeys' => true,
|
||
|
'nopermissions' => true
|
||
|
]);
|
||
|
|
||
|
$items_to_delete = array_diff_key($db_itemids, $itemids);
|
||
|
|
||
|
if ($items_to_delete) {
|
||
|
API::Item()->delete(array_keys($items_to_delete));
|
||
|
}
|
||
|
|
||
|
$this->referencer->refreshItems();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Deletes triggers from DB that are missing in import file.
|
||
|
*/
|
||
|
protected function deleteMissingTriggers(): void {
|
||
|
if (!$this->options['triggers']['deleteMissing']) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$processed_hostids = array_merge(
|
||
|
$this->importedObjectContainer->getHostids(),
|
||
|
$this->importedObjectContainer->getTemplateids()
|
||
|
);
|
||
|
|
||
|
if (!$processed_hostids) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$triggerids = [];
|
||
|
|
||
|
foreach ($this->getFormattedTriggers() as $trigger) {
|
||
|
$triggerid = null;
|
||
|
|
||
|
if (array_key_exists('uuid', $trigger)) {
|
||
|
$triggerid = $this->referencer->findTriggeridByUuid($trigger['uuid']);
|
||
|
}
|
||
|
|
||
|
// In import file host trigger can have UUID assigned after conversion, such should be searched by name.
|
||
|
if ($triggerid === null) {
|
||
|
$triggerid = $this->referencer->findTriggeridByName($trigger['description'], $trigger['expression'],
|
||
|
$trigger['recovery_expression']
|
||
|
);
|
||
|
|
||
|
// Template triggers should only be searched by UUID.
|
||
|
if ($triggerid !== null && array_key_exists('uuid', $trigger)) {
|
||
|
$db_trigger = $this->referencer->findTriggerById($triggerid);
|
||
|
|
||
|
if ($db_trigger['uuid'] !== '' && $db_trigger['uuid'] !== $trigger['uuid']) {
|
||
|
$triggerid = null;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($triggerid !== null) {
|
||
|
$triggerids[$triggerid] = [];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$db_triggerids = API::Trigger()->get([
|
||
|
'output' => [],
|
||
|
'selectHosts' => ['hostid'],
|
||
|
'hostids' => $processed_hostids,
|
||
|
'filter' => ['flags' => ZBX_FLAG_DISCOVERY_NORMAL],
|
||
|
'inherited' => false,
|
||
|
'preservekeys' => true,
|
||
|
'nopermissions' => true
|
||
|
]);
|
||
|
|
||
|
// Check that potentially deletable trigger belongs to same hosts that are in the import file.
|
||
|
// If some triggers belong to more hosts than import file contains, don't delete them.
|
||
|
$triggerids_to_delete = [];
|
||
|
$processed_hostids = array_flip($processed_hostids);
|
||
|
|
||
|
foreach (array_diff_key($db_triggerids, $triggerids) as $triggerid => $trigger) {
|
||
|
$trigger_hostids = array_flip(array_column($trigger['hosts'], 'hostid'));
|
||
|
|
||
|
if (!array_diff_key($trigger_hostids, $processed_hostids)) {
|
||
|
$triggerids_to_delete[] = $triggerid;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($triggerids_to_delete) {
|
||
|
API::Trigger()->delete($triggerids_to_delete);
|
||
|
}
|
||
|
|
||
|
// refresh triggers because template triggers can be inherited to host and used in maps
|
||
|
$this->referencer->refreshTriggers();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Deletes graphs from DB that are missing in import file.
|
||
|
*/
|
||
|
protected function deleteMissingGraphs(): void {
|
||
|
if (!$this->options['graphs']['deleteMissing']) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$processed_hostids = array_merge(
|
||
|
$this->importedObjectContainer->getHostids(),
|
||
|
$this->importedObjectContainer->getTemplateids()
|
||
|
);
|
||
|
|
||
|
if (!$processed_hostids) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$graphids = [];
|
||
|
|
||
|
foreach ($this->getFormattedGraphs() as $graph) {
|
||
|
$graphid = null;
|
||
|
|
||
|
if (array_key_exists('uuid', $graph)) {
|
||
|
$graphid = $this->referencer->findGraphidByUuid($graph['uuid']);
|
||
|
}
|
||
|
|
||
|
if ($graphid !== null) {
|
||
|
$graphids[$graphid] = [];
|
||
|
}
|
||
|
elseif (array_key_exists('gitems', $graph)) {
|
||
|
// In import file host graph can have UUID assigned after conversion, such should be searched by name.
|
||
|
foreach ($graph['gitems'] as $gitem) {
|
||
|
$gitem_hostid = $this->referencer->findTemplateidOrHostidByHost($gitem['item']['host']);
|
||
|
|
||
|
if ($gitem_hostid !== null) {
|
||
|
$graphid = $this->referencer->findGraphidByName($gitem_hostid, $graph['name']);
|
||
|
}
|
||
|
|
||
|
if ($graphid !== null) {
|
||
|
$graphids[$graphid] = [];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$db_graphids = API::Graph()->get([
|
||
|
'output' => ['graphid'],
|
||
|
'hostids' => $processed_hostids,
|
||
|
'selectHosts' => ['hostid'],
|
||
|
'preservekeys' => true,
|
||
|
'nopermissions' => true,
|
||
|
'inherited' => false,
|
||
|
'filter' => ['flags' => ZBX_FLAG_DISCOVERY_NORMAL]
|
||
|
]);
|
||
|
|
||
|
// check that potentially deletable graph belongs to same hosts that are in XML
|
||
|
// if some graphs belong to more hosts than current XML contains, don't delete them
|
||
|
$graphids_to_delete = [];
|
||
|
$processed_hostids = array_flip($processed_hostids);
|
||
|
|
||
|
foreach (array_diff_key($db_graphids, $graphids) as $graphid => $graph) {
|
||
|
$graph_hostids = array_flip(array_column($graph['hosts'], 'hostid'));
|
||
|
|
||
|
if (!array_diff_key($graph_hostids, $processed_hostids)) {
|
||
|
$graphids_to_delete[] = $graphid;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($graphids_to_delete) {
|
||
|
API::Graph()->delete($graphids_to_delete);
|
||
|
}
|
||
|
|
||
|
$this->referencer->refreshGraphs();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Deletes prototypes from DB that are missing in import file.
|
||
|
*
|
||
|
* @param array $discovery_rules_by_hosts
|
||
|
*
|
||
|
* @throws APIException
|
||
|
*/
|
||
|
protected function deleteMissingPrototypes(array $discovery_rules_by_hosts): void {
|
||
|
$discovery_ruleids = [];
|
||
|
$host_prototypeids = [];
|
||
|
$trigger_prototypeids = [];
|
||
|
$graph_prototypeids = [];
|
||
|
$item_prototypeids = [];
|
||
|
|
||
|
foreach ($discovery_rules_by_hosts as $host => $discovery_rules) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($host);
|
||
|
|
||
|
foreach ($discovery_rules as $discovery_rule) {
|
||
|
$discoveryid = null;
|
||
|
|
||
|
if (array_key_exists('uuid', $discovery_rule)) {
|
||
|
$discoveryid = $this->referencer->findItemidByUuid($discovery_rule['uuid']);
|
||
|
}
|
||
|
|
||
|
if ($discoveryid === null) {
|
||
|
$discoveryid = $this->referencer->findItemidByKey($hostid, $discovery_rule['key_']);
|
||
|
}
|
||
|
|
||
|
if ($discoveryid === null) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$discovery_ruleids[$discoveryid] = [];
|
||
|
|
||
|
foreach ($discovery_rule['host_prototypes'] as $host_prototype) {
|
||
|
$host_prototypeid = null;
|
||
|
|
||
|
if (array_key_exists('uuid', $host_prototype)) {
|
||
|
$host_prototypeid = $this->referencer->findHostPrototypeidByUuid($host_prototype['uuid']);
|
||
|
}
|
||
|
|
||
|
if ($host_prototypeid === null) {
|
||
|
$host_prototypeid = $this->referencer->findHostPrototypeidByHost($hostid, $discoveryid,
|
||
|
$host_prototype['host']
|
||
|
);
|
||
|
}
|
||
|
|
||
|
if ($host_prototypeid !== null) {
|
||
|
$host_prototypeids[$host_prototypeid] = [];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($discovery_rule['trigger_prototypes'] as $trigger_prototype) {
|
||
|
$trigger_prototypeid = null;
|
||
|
|
||
|
if (array_key_exists('uuid', $trigger_prototype)) {
|
||
|
$trigger_prototypeid = $this->referencer->findTriggeridByUuid($trigger_prototype['uuid']);
|
||
|
}
|
||
|
|
||
|
if ($trigger_prototypeid === null) {
|
||
|
$trigger_prototypeid = $this->referencer->findTriggeridByName($trigger_prototype['description'],
|
||
|
$trigger_prototype['expression'], $trigger_prototype['recovery_expression']
|
||
|
);
|
||
|
}
|
||
|
|
||
|
if ($trigger_prototypeid !== null) {
|
||
|
$trigger_prototypeids[$trigger_prototypeid] = [];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($discovery_rule['graph_prototypes'] as $graph_prototype) {
|
||
|
$graph_prototypeid = null;
|
||
|
|
||
|
if (array_key_exists('uuid', $graph_prototype)) {
|
||
|
$graph_prototypeid = $this->referencer->findGraphidByUuid($graph_prototype['uuid']);
|
||
|
}
|
||
|
|
||
|
if ($graph_prototypeid === null) {
|
||
|
$graph_prototypeid = $this->referencer->findGraphidByName($hostid, $graph_prototype['name']);
|
||
|
}
|
||
|
|
||
|
if ($graph_prototypeid !== null) {
|
||
|
$graph_prototypeids[$graph_prototypeid] = [];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
foreach ($discovery_rule['item_prototypes'] as $item_prototype) {
|
||
|
$item_prototypeid = null;
|
||
|
|
||
|
if (array_key_exists('uuid', $item_prototype)) {
|
||
|
$item_prototypeid = $this->referencer->findItemidByUuid($item_prototype['uuid']);
|
||
|
}
|
||
|
|
||
|
if ($item_prototypeid === null) {
|
||
|
$item_prototypeid = $this->referencer->findItemidByKey($hostid, $item_prototype['key_']);
|
||
|
}
|
||
|
|
||
|
if ($item_prototypeid !== null) {
|
||
|
$item_prototypeids[$item_prototypeid] = [];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$db_host_prototypes = API::HostPrototype()->get([
|
||
|
'output' => [],
|
||
|
'discoveryids' => array_keys($discovery_ruleids),
|
||
|
'preservekeys' => true,
|
||
|
'nopermissions' => true,
|
||
|
'inherited' => false
|
||
|
]);
|
||
|
|
||
|
$host_prototypes_to_delete = array_diff_key($db_host_prototypes, $host_prototypeids);
|
||
|
|
||
|
if ($host_prototypes_to_delete) {
|
||
|
API::HostPrototype()->delete(array_keys($host_prototypes_to_delete));
|
||
|
}
|
||
|
|
||
|
$db_trigger_prototypes = API::TriggerPrototype()->get([
|
||
|
'output' => [],
|
||
|
'discoveryids' => array_keys($discovery_ruleids),
|
||
|
'preservekeys' => true,
|
||
|
'nopermissions' => true,
|
||
|
'inherited' => false
|
||
|
]);
|
||
|
|
||
|
$trigger_prototypes_to_delete = array_diff_key($db_trigger_prototypes, $trigger_prototypeids);
|
||
|
|
||
|
// Unlike triggers that belong to multiple hosts, trigger prototypes do not, so we just delete them.
|
||
|
if ($trigger_prototypes_to_delete) {
|
||
|
API::TriggerPrototype()->delete(array_keys($trigger_prototypes_to_delete));
|
||
|
|
||
|
$this->referencer->refreshTriggers();
|
||
|
}
|
||
|
|
||
|
$db_graph_prototypes = API::GraphPrototype()->get([
|
||
|
'output' => [],
|
||
|
'discoveryids' => array_keys($discovery_ruleids),
|
||
|
'preservekeys' => true,
|
||
|
'nopermissions' => true,
|
||
|
'inherited' => false
|
||
|
]);
|
||
|
|
||
|
$graph_prototypes_to_delete = array_diff_key($db_graph_prototypes, $graph_prototypeids);
|
||
|
|
||
|
// Unlike graphs that belong to multiple hosts, graph prototypes do not, so we just delete them.
|
||
|
if ($graph_prototypes_to_delete) {
|
||
|
API::GraphPrototype()->delete(array_keys($graph_prototypes_to_delete));
|
||
|
|
||
|
$this->referencer->refreshGraphs();
|
||
|
}
|
||
|
|
||
|
$db_item_prototypes = API::ItemPrototype()->get([
|
||
|
'output' => [],
|
||
|
'discoveryids' => array_keys($discovery_ruleids),
|
||
|
'preservekeys' => true,
|
||
|
'nopermissions' => true,
|
||
|
'inherited' => false
|
||
|
]);
|
||
|
|
||
|
$item_prototypes_to_delete = array_diff_key($db_item_prototypes, $item_prototypeids);
|
||
|
|
||
|
if ($item_prototypes_to_delete) {
|
||
|
API::ItemPrototype()->delete(array_keys($item_prototypes_to_delete));
|
||
|
|
||
|
$this->referencer->refreshItems();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Deletes web scenarios from DB that are missing in import file.
|
||
|
*/
|
||
|
protected function deleteMissingHttpTests(): void {
|
||
|
if (!$this->options['httptests']['deleteMissing']) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$processed_hostids = array_merge(
|
||
|
$this->importedObjectContainer->getHostids(),
|
||
|
$this->importedObjectContainer->getTemplateids()
|
||
|
);
|
||
|
|
||
|
if (!$processed_hostids) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$httptestids = [];
|
||
|
|
||
|
foreach ($this->getFormattedHttpTests() as $host => $httptests) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($host);
|
||
|
|
||
|
if ($hostid === null) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
foreach ($httptests as $httptest) {
|
||
|
$httptestid = null;
|
||
|
|
||
|
if (array_key_exists('uuid', $httptest)) {
|
||
|
$httptestid = $this->referencer->findHttpTestidByUuid($httptest['uuid']);
|
||
|
}
|
||
|
|
||
|
if ($httptestid === null) {
|
||
|
$httptestid = $this->referencer->findHttpTestidByName($hostid, $httptest['name']);
|
||
|
}
|
||
|
|
||
|
if ($httptestid !== null) {
|
||
|
$httptestids[$httptestid] = [];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$db_httptestids = API::HttpTest()->get([
|
||
|
'output' => [],
|
||
|
'hostids' => $processed_hostids,
|
||
|
'inherited' => false,
|
||
|
'preservekeys' => true,
|
||
|
'nopermissions' => true
|
||
|
]);
|
||
|
|
||
|
$httptestids_to_delete = array_diff_key($db_httptestids, $httptestids);
|
||
|
|
||
|
if ($httptestids_to_delete) {
|
||
|
API::HttpTest()->delete(array_keys($httptestids_to_delete));
|
||
|
}
|
||
|
|
||
|
$this->referencer->refreshHttpTests();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Deletes template dashboards from DB that are missing in import file.
|
||
|
*
|
||
|
* @throws APIException
|
||
|
*/
|
||
|
protected function deleteMissingTemplateDashboards(): void {
|
||
|
if (!$this->options['templateDashboards']['deleteMissing']) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$dashboard_importer = new CTemplateDashboardImporter($this->options, $this->referencer,
|
||
|
$this->importedObjectContainer
|
||
|
);
|
||
|
|
||
|
$dashboard_importer->delete($this->getFormattedTemplateDashboards());
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Deletes discovery rules from DB that are missing in import file.
|
||
|
*/
|
||
|
protected function deleteMissingDiscoveryRules(): void {
|
||
|
if (!$this->options['discoveryRules']['deleteMissing']) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$processed_hostids = array_merge(
|
||
|
$this->importedObjectContainer->getHostids(),
|
||
|
$this->importedObjectContainer->getTemplateids()
|
||
|
);
|
||
|
|
||
|
if (!$processed_hostids) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$discovery_ruleids = [];
|
||
|
|
||
|
foreach ($this->getFormattedDiscoveryRules() as $host => $discovery_rules) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($host);
|
||
|
|
||
|
if ($hostid === null) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
foreach ($discovery_rules as $discovery_rule) {
|
||
|
$discovery_ruleid = null;
|
||
|
|
||
|
if (array_key_exists('uuid', $discovery_rule)) {
|
||
|
$discovery_ruleid = $this->referencer->findItemidByUuid($discovery_rule['uuid']);
|
||
|
}
|
||
|
|
||
|
if ($discovery_ruleid === null) {
|
||
|
$discovery_ruleid = $this->referencer->findItemidByKey($hostid, $discovery_rule['key_']);
|
||
|
}
|
||
|
|
||
|
if ($discovery_ruleid !== null) {
|
||
|
$discovery_ruleids[$discovery_ruleid] = [];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$db_discovery_ruleids = API::DiscoveryRule()->get([
|
||
|
'output' => ['itemid'],
|
||
|
'hostids' => $processed_hostids,
|
||
|
'inherited' => false,
|
||
|
'preservekeys' => true,
|
||
|
'nopermissions' => true
|
||
|
]);
|
||
|
|
||
|
$discovery_ruleids_to_delete = array_diff_key($db_discovery_ruleids, $discovery_ruleids);
|
||
|
|
||
|
if ($discovery_ruleids_to_delete) {
|
||
|
API::DiscoveryRule()->delete(array_keys($discovery_ruleids_to_delete));
|
||
|
}
|
||
|
|
||
|
$this->referencer->refreshItems();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get formatted template groups.
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
protected function getFormattedTemplateGroups(): array {
|
||
|
if (!array_key_exists('template_groups', $this->formattedData)) {
|
||
|
$this->formattedData['template_groups'] = $this->adapter->getTemplateGroups();
|
||
|
}
|
||
|
|
||
|
return $this->formattedData['template_groups'];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get formatted host groups.
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
protected function getFormattedHostGroups(): array {
|
||
|
if (!array_key_exists('host_groups', $this->formattedData)) {
|
||
|
$this->formattedData['host_groups'] = $this->adapter->getHostGroups();
|
||
|
}
|
||
|
|
||
|
return $this->formattedData['host_groups'];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get formatted templates.
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
public function getFormattedTemplates(): array {
|
||
|
if (!array_key_exists('templates', $this->formattedData)) {
|
||
|
$this->formattedData['templates'] = $this->adapter->getTemplates();
|
||
|
}
|
||
|
|
||
|
return $this->formattedData['templates'];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get formatted hosts.
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
public function getFormattedHosts(): array {
|
||
|
if (!array_key_exists('hosts', $this->formattedData)) {
|
||
|
$this->formattedData['hosts'] = $this->adapter->getHosts();
|
||
|
}
|
||
|
|
||
|
return $this->formattedData['hosts'];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get formatted items.
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
protected function getFormattedItems(): array {
|
||
|
if (!array_key_exists('items', $this->formattedData)) {
|
||
|
$this->formattedData['items'] = $this->adapter->getItems();
|
||
|
}
|
||
|
|
||
|
return $this->formattedData['items'];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get formatted discovery rules.
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
protected function getFormattedDiscoveryRules(): array {
|
||
|
if (!array_key_exists('discoveryRules', $this->formattedData)) {
|
||
|
$this->formattedData['discoveryRules'] = $this->adapter->getDiscoveryRules();
|
||
|
}
|
||
|
|
||
|
return $this->formattedData['discoveryRules'];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get formatted web scenarios.
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
protected function getFormattedHttpTests(): array {
|
||
|
if (!array_key_exists('httptests', $this->formattedData)) {
|
||
|
$this->formattedData['httptests'] = $this->adapter->getHttpTests();
|
||
|
}
|
||
|
|
||
|
return $this->formattedData['httptests'];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get formatted web scenario steps.
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
protected function getFormattedHttpSteps(): array {
|
||
|
if (!array_key_exists('httpsteps', $this->formattedData)) {
|
||
|
$this->formattedData['httpsteps'] = $this->adapter->getHttpSteps();
|
||
|
}
|
||
|
|
||
|
return $this->formattedData['httpsteps'];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get formatted triggers.
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
protected function getFormattedTriggers(): array {
|
||
|
if (!array_key_exists('triggers', $this->formattedData)) {
|
||
|
$this->formattedData['triggers'] = $this->adapter->getTriggers();
|
||
|
}
|
||
|
|
||
|
return $this->formattedData['triggers'];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get formatted graphs.
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
protected function getFormattedGraphs(): array {
|
||
|
if (!array_key_exists('graphs', $this->formattedData)) {
|
||
|
$this->formattedData['graphs'] = $this->adapter->getGraphs();
|
||
|
}
|
||
|
|
||
|
return $this->formattedData['graphs'];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get formatted images.
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
protected function getFormattedImages(): array {
|
||
|
if (!array_key_exists('images', $this->formattedData)) {
|
||
|
$this->formattedData['images'] = $this->adapter->getImages();
|
||
|
}
|
||
|
|
||
|
return $this->formattedData['images'];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get formatted maps.
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
protected function getFormattedMaps(): array {
|
||
|
if (!array_key_exists('maps', $this->formattedData)) {
|
||
|
$this->formattedData['maps'] = $this->adapter->getMaps();
|
||
|
}
|
||
|
|
||
|
return $this->formattedData['maps'];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get formatted template dashboards.
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
protected function getFormattedTemplateDashboards(): array {
|
||
|
if (!array_key_exists('templateDashboards', $this->formattedData)) {
|
||
|
$this->formattedData['templateDashboards'] = $this->adapter->getTemplateDashboards();
|
||
|
}
|
||
|
|
||
|
return $this->formattedData['templateDashboards'];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get formatted media types.
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
protected function getFormattedMediaTypes(): array {
|
||
|
if (!array_key_exists('mediaTypes', $this->formattedData)) {
|
||
|
$this->formattedData['mediaTypes'] = $this->adapter->getMediaTypes();
|
||
|
}
|
||
|
|
||
|
return $this->formattedData['mediaTypes'];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get items keys order tree, to ensure that master item will be inserted or updated before any of it dependent
|
||
|
* item. Returns associative array where key is item index and value is item dependency level.
|
||
|
*
|
||
|
* @param string $master_item_key String containing master key name used to identify item master.
|
||
|
*
|
||
|
* @return array
|
||
|
*
|
||
|
* @throws Exception
|
||
|
*/
|
||
|
protected function getItemsOrder(string $master_item_key): array {
|
||
|
$entities = $this->getFormattedItems();
|
||
|
|
||
|
return $this->getEntitiesOrder($entities, $master_item_key);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get discovery rules items prototypes keys order tree, to ensure that master item will be inserted or updated
|
||
|
* before any of it dependent item. Returns associative array where key is item prototype index and value is item
|
||
|
* prototype dependency level.
|
||
|
*
|
||
|
* @param string $master_item_key String containing master key name used to identify item master.
|
||
|
*
|
||
|
* @return array
|
||
|
*
|
||
|
* @throws Exception
|
||
|
*/
|
||
|
protected function getDiscoveryRulesItemsOrder(string $master_item_key): array {
|
||
|
$discovery_rules_by_hosts = $this->getFormattedDiscoveryRules();
|
||
|
$entities_order = [];
|
||
|
|
||
|
foreach ($discovery_rules_by_hosts as $host => $discovery_rules) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($host);
|
||
|
|
||
|
if ($hostid === null) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
foreach ($discovery_rules as $discovery_rule) {
|
||
|
if ($discovery_rule['item_prototypes']) {
|
||
|
$item_prototypes = [$host => $discovery_rule['item_prototypes']];
|
||
|
$item_prototypes = $this->getEntitiesOrder($item_prototypes, $master_item_key, true);
|
||
|
$entities_order[$host][$discovery_rule['key_']] = $item_prototypes[$host];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $entities_order;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Generic method to get entities order tree, to ensure that master entity will be inserted or updated before any
|
||
|
* of it dependent entities.
|
||
|
* Returns associative array where key is entity index in source array grouped by host key and value is entity
|
||
|
* dependency level.
|
||
|
*
|
||
|
* @param array $items_by_hosts Associative array of host key and host items.
|
||
|
* @param string $master_item_key String containing master key name to identify item master.
|
||
|
* @param bool $get_prototypes Option to get also master item prototypes not found in supplied input.
|
||
|
*
|
||
|
* @return array
|
||
|
*
|
||
|
* @throws Exception if data is invalid.
|
||
|
*/
|
||
|
protected function getEntitiesOrder(array $items_by_hosts, string $master_item_key,
|
||
|
bool $get_prototypes = false): array {
|
||
|
$parent_item_hostids = [];
|
||
|
$parent_item_keys = [];
|
||
|
$resolved_masters_cache = [];
|
||
|
|
||
|
$host_name_to_hostid = [];
|
||
|
|
||
|
foreach ($items_by_hosts as $host_name => $items) {
|
||
|
$hostid = $this->referencer->findTemplateidOrHostidByHost($host_name);
|
||
|
|
||
|
if ($hostid === null) {
|
||
|
unset($items_by_hosts[$host_name]);
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$host_name_to_hostid[$host_name] = $hostid;
|
||
|
|
||
|
// Cache input array entities.
|
||
|
foreach ($items as $item) {
|
||
|
$resolved_masters_cache[$host_name][$item['key_']] = [
|
||
|
'type' => $item['type'],
|
||
|
$master_item_key => $item[$master_item_key]
|
||
|
];
|
||
|
|
||
|
if ($item['type'] == ITEM_TYPE_DEPENDENT && array_key_exists('key', $item[$master_item_key])) {
|
||
|
$parent_item_hostids[$hostid] = true;
|
||
|
$parent_item_keys[$item[$master_item_key]['key']] = true;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// There are entities to resolve from database, resolve and cache them recursively.
|
||
|
if ($parent_item_keys) {
|
||
|
/*
|
||
|
* For existing items, 'referencer' should be initialized before 'setDbItem' method will be used.
|
||
|
* Registering reference when property 'db_items' is empty, will not allow first call of
|
||
|
* 'findValueMapidByName' method update references to existing items.
|
||
|
*/
|
||
|
$this->referencer->initItemsReferences();
|
||
|
|
||
|
$options = [
|
||
|
'output' => ['itemid', 'uuid', 'key_', 'type', 'hostid', 'master_itemid'],
|
||
|
'hostids' => array_keys($parent_item_hostids),
|
||
|
'filter' => ['key_' => array_keys($parent_item_keys)],
|
||
|
'preservekeys' => true
|
||
|
];
|
||
|
|
||
|
$db_items = API::Item()->get($options + ['webitems' => true]);
|
||
|
|
||
|
if ($get_prototypes) {
|
||
|
$db_items += API::ItemPrototype()->get($options);
|
||
|
}
|
||
|
|
||
|
$resolve_entity_keys = [];
|
||
|
$itemid_to_item_key_by_hosts = [];
|
||
|
|
||
|
for ($level = 0; $level < ZBX_DEPENDENT_ITEM_MAX_LEVELS; $level++) {
|
||
|
$missing_master_itemids = [];
|
||
|
|
||
|
foreach ($db_items as $itemid => $item) {
|
||
|
$host_name = array_search($item['hostid'], $host_name_to_hostid);
|
||
|
|
||
|
$this->referencer->setDbItem($itemid, $item);
|
||
|
|
||
|
$item['key'] = $item['key_'];
|
||
|
unset($item['key_']);
|
||
|
|
||
|
$itemid_to_item_key_by_hosts[$host_name][$itemid] = $item['key'];
|
||
|
|
||
|
$cache_entity = [
|
||
|
'type' => $item['type']
|
||
|
];
|
||
|
|
||
|
if ($item['type'] == ITEM_TYPE_DEPENDENT) {
|
||
|
$master_itemid = $item['master_itemid'];
|
||
|
|
||
|
if (array_key_exists($master_itemid, $itemid_to_item_key_by_hosts[$host_name])) {
|
||
|
$cache_entity[$master_item_key] = [
|
||
|
'key' => $itemid_to_item_key_by_hosts[$host_name][$master_itemid]
|
||
|
];
|
||
|
}
|
||
|
else {
|
||
|
$missing_master_itemids[] = $item['master_itemid'];
|
||
|
$resolve_entity_keys[] = [
|
||
|
'host' => $host_name,
|
||
|
'key' => $item['key'],
|
||
|
'master_itemid' => $item['master_itemid']
|
||
|
];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$resolved_masters_cache[$host_name][$item['key']] = $cache_entity;
|
||
|
}
|
||
|
|
||
|
if ($missing_master_itemids) {
|
||
|
$options = [
|
||
|
'output' => ['uuid', 'key_', 'type', 'hostid', 'master_itemid'],
|
||
|
'itemids' => $missing_master_itemids,
|
||
|
'preservekeys' => true
|
||
|
];
|
||
|
$db_items = API::Item()->get($options + ['webitems' => true]);
|
||
|
|
||
|
if ($get_prototypes) {
|
||
|
$db_items += API::ItemPrototype()->get($options);
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($missing_master_itemids) {
|
||
|
throw new Exception(_s('Incorrect value for field "%1$s": %2$s.', 'master_itemid',
|
||
|
_('maximum number of dependency levels reached')
|
||
|
));
|
||
|
}
|
||
|
|
||
|
foreach ($resolve_entity_keys as $item) {
|
||
|
$master_itemid = $item['master_itemid'];
|
||
|
|
||
|
if (!array_key_exists($item['host'], $itemid_to_item_key_by_hosts) ||
|
||
|
!array_key_exists($master_itemid, $itemid_to_item_key_by_hosts[$item['host']])) {
|
||
|
throw new Exception(_s('Incorrect value for field "%1$s": %2$s.', 'master_itemid',
|
||
|
_s('value "%1$s" not found', $master_itemid)
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$master_key = $itemid_to_item_key_by_hosts[$item['host']][$master_itemid];
|
||
|
$resolved_masters_cache[$item['host']][$item['key']] += [
|
||
|
$master_item_key => ['key' => $master_key]
|
||
|
];
|
||
|
}
|
||
|
|
||
|
unset($resolve_entity_keys, $itemid_to_item_key_by_hosts);
|
||
|
}
|
||
|
|
||
|
// Resolve every entity dependency level.
|
||
|
$tree = [];
|
||
|
|
||
|
foreach ($items_by_hosts as $host_name => $items) {
|
||
|
$hostid = $host_name_to_hostid[$host_name];
|
||
|
$host_items_tree = [];
|
||
|
|
||
|
foreach ($items as $index => $item) {
|
||
|
$level = 0;
|
||
|
$traversal_path = [$item['key_']];
|
||
|
|
||
|
while ($item && $item['type'] == ITEM_TYPE_DEPENDENT) {
|
||
|
if (!array_key_exists('key', $item[$master_item_key])) {
|
||
|
throw new Exception(_s('Incorrect value for field "%1$s": %2$s.', 'master_itemid',
|
||
|
_('cannot be empty')
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$master_key = $item[$master_item_key]['key'];
|
||
|
|
||
|
if (array_key_exists($host_name, $resolved_masters_cache)
|
||
|
&& array_key_exists($master_key, $resolved_masters_cache[$host_name])) {
|
||
|
$item = $resolved_masters_cache[$host_name][$master_key];
|
||
|
|
||
|
if (($item['type'] == ITEM_TYPE_DEPENDENT
|
||
|
&& $item[$master_item_key]
|
||
|
&& $master_key === $item[$master_item_key]['key'])
|
||
|
|| in_array($master_key, $traversal_path)) {
|
||
|
throw new Exception(_s('Incorrect value for field "%1$s": %2$s.', 'master_itemid',
|
||
|
_('circular item dependency is not allowed')
|
||
|
));
|
||
|
}
|
||
|
|
||
|
$traversal_path[] = $master_key;
|
||
|
$level++;
|
||
|
}
|
||
|
else {
|
||
|
throw new Exception(_s('Incorrect value for field "%1$s": %2$s.', 'master_itemid',
|
||
|
_s('value "%1$s" not found', $master_key)
|
||
|
));
|
||
|
}
|
||
|
|
||
|
if ($level > ZBX_DEPENDENT_ITEM_MAX_LEVELS) {
|
||
|
throw new Exception(_s('Incorrect value for field "%1$s": %2$s.', 'master_itemid',
|
||
|
_('maximum number of dependency levels reached')
|
||
|
));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$host_items_tree[$index] = $level;
|
||
|
}
|
||
|
|
||
|
$tree[$host_name] = $host_items_tree;
|
||
|
}
|
||
|
|
||
|
// Order item indexes in descending order by nesting level.
|
||
|
foreach ($tree as &$item_indexes) {
|
||
|
asort($item_indexes);
|
||
|
}
|
||
|
unset($item_indexes);
|
||
|
|
||
|
return $tree;
|
||
|
}
|
||
|
}
|