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.

1495 lines
47 KiB

<?php declare(strict_types = 0);
/*
** 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.
**/
/**
* SLA API implementation.
*/
class CSla extends CApiService {
public const ACCESS_RULES = [
'get' => ['min_user_type' => USER_TYPE_ZABBIX_USER],
'getsli' => ['min_user_type' => USER_TYPE_ZABBIX_USER],
'create' => ['min_user_type' => USER_TYPE_ZABBIX_ADMIN, 'action' => CRoleHelper::ACTIONS_MANAGE_SLA],
'update' => ['min_user_type' => USER_TYPE_ZABBIX_ADMIN, 'action' => CRoleHelper::ACTIONS_MANAGE_SLA],
'delete' => ['min_user_type' => USER_TYPE_ZABBIX_ADMIN, 'action' => CRoleHelper::ACTIONS_MANAGE_SLA]
];
protected $tableName = 'sla';
protected $tableAlias = 'sla';
protected $sortColumns = ['slaid', 'name', 'period', 'slo', 'effective_date', 'timezone', 'status', 'description'];
/**
* @param array $options
*
* @throws APIException
*
* @return array|string
*/
public function get(array $options = []) {
$api_input_rules = ['type' => API_OBJECT, 'fields' => [
// filter
'slaids' => ['type' => API_IDS, 'flags' => API_ALLOW_NULL | API_NORMALIZE, 'default' => null],
'evaltype' => ['type' => API_INT32, 'in' => implode(',', [TAG_EVAL_TYPE_AND_OR, TAG_EVAL_TYPE_OR]), 'default' => TAG_EVAL_TYPE_AND_OR],
'service_tags' => ['type' => API_OBJECTS, 'default' => [], 'fields' => [
'tag' => ['type' => API_STRING_UTF8, 'flags' => API_REQUIRED],
'value' => ['type' => API_STRING_UTF8],
'operator' => ['type' => API_INT32, 'in' => implode(',', [TAG_OPERATOR_LIKE, TAG_OPERATOR_EQUAL, TAG_OPERATOR_NOT_LIKE, TAG_OPERATOR_NOT_EQUAL, TAG_OPERATOR_EXISTS, TAG_OPERATOR_NOT_EXISTS])]
]],
'serviceids' => ['type' => API_IDS, 'flags' => API_ALLOW_NULL | API_NORMALIZE, 'default' => null],
'filter' => ['type' => API_FILTER, 'flags' => API_ALLOW_NULL, 'default' => null, 'fields' => ['slaid', 'name', 'period', 'slo', 'effective_date', 'timezone', 'status']],
'search' => ['type' => API_FILTER, 'flags' => API_ALLOW_NULL, 'default' => null, 'fields' => ['name', 'timezone', 'description']],
'searchByAny' => ['type' => API_BOOLEAN, 'default' => false],
'startSearch' => ['type' => API_FLAG, 'default' => false],
'excludeSearch' => ['type' => API_FLAG, 'default' => false],
'searchWildcardsEnabled' => ['type' => API_BOOLEAN, 'default' => false],
// output
'output' => ['type' => API_OUTPUT, 'in' => implode(',', ['slaid', 'name', 'period', 'slo', 'effective_date', 'timezone', 'status', 'description']), 'default' => API_OUTPUT_EXTEND],
'countOutput' => ['type' => API_FLAG, 'default' => false],
'selectServiceTags' => ['type' => API_OUTPUT, 'flags' => API_ALLOW_NULL | API_ALLOW_COUNT, 'in' => implode(',', ['tag', 'operator', 'value']), 'default' => null],
'selectSchedule' => ['type' => API_OUTPUT, 'flags' => API_ALLOW_NULL | API_ALLOW_COUNT, 'in' => implode(',', ['period_from', 'period_to']), 'default' => null],
'selectExcludedDowntimes' => ['type' => API_OUTPUT, 'flags' => API_ALLOW_NULL | API_ALLOW_COUNT, 'in' => implode(',', ['name', 'period_from', 'period_to']), 'default' => null],
// sort and limit
'sortfield' => ['type' => API_STRINGS_UTF8, 'flags' => API_NORMALIZE, 'in' => implode(',', ['slaid', 'name', 'period', 'slo', 'effective_date', 'timezone', 'status', 'description']), 'uniq' => true, 'default' => []],
'sortorder' => ['type' => API_SORTORDER, 'default' => []],
'limit' => ['type' => API_INT32, 'flags' => API_ALLOW_NULL, 'in' => '1:'.ZBX_MAX_INT32, 'default' => null],
// flags
'editable' => ['type' => API_BOOLEAN, 'default' => false],
'preservekeys' => ['type' => API_BOOLEAN, 'default' => false]
]];
if (!CApiInputValidator::validate($api_input_rules, $options, '/', $error)) {
self::exception(ZBX_API_ERROR_PARAMETERS, $error);
}
$accessible_slaids = self::getAccessibleSlaids($options['slaids'], $options['serviceids']);
$count_output = $options['countOutput'];
if ($count_output) {
$options['output'] = ['slaid'];
$options['countOutput'] = false;
}
$resource = DBselect($this->createSelectQuery('sla', $options));
$db_slas = [];
while (($options['limit'] === null || count($db_slas) < $options['limit']) && $row = DBfetch($resource)) {
if ($accessible_slaids !== null && !array_key_exists($row['slaid'], $accessible_slaids)) {
continue;
}
$db_slas[$row['slaid']] = $row;
}
if ($count_output) {
return (string) count($db_slas);
}
if ($db_slas) {
$db_slas = $this->addRelatedObjects($options, $db_slas);
$db_slas = $this->unsetExtraFields($db_slas, ['slaid'], $options['output']);
if (!$options['preservekeys']) {
$db_slas = array_values($db_slas);
}
}
return $db_slas;
}
/**
* @param array $slas
*
* @throws APIException
*
* @return array
*/
public function create(array $slas): array {
self::validateCreate($slas);
$ins_slas = [];
foreach ($slas as $sla) {
unset($sla['service_tags'], $sla['schedule'], $sla['excluded_downtimes']);
$ins_slas[] = $sla;
}
$slaids = DB::insert('sla', $ins_slas);
foreach ($slas as $index => &$sla) {
$sla['slaid'] = $slaids[$index];
}
unset($sla);
self::updateServiceTags($slas);
self::updateSchedule($slas);
self::updateExcludedDowntimes($slas);
self::addAuditLog(CAudit::ACTION_ADD, CAudit::RESOURCE_SLA, $slas);
return ['slaids' => $slaids];
}
/**
* @param array $slas
*
* @throws APIException
*/
private static function validateCreate(array &$slas): void {
$api_input_rules = ['type' => API_OBJECTS, 'flags' => API_NOT_EMPTY | API_NORMALIZE, 'uniq' => [['name']], 'fields' => [
'name' => ['type' => API_STRING_UTF8, 'flags' => API_REQUIRED | API_NOT_EMPTY, 'length' => DB::getFieldLength('sla', 'name')],
'period' => ['type' => API_INT32, 'flags' => API_REQUIRED, 'in' => implode(',', [ZBX_SLA_PERIOD_DAILY, ZBX_SLA_PERIOD_WEEKLY, ZBX_SLA_PERIOD_MONTHLY, ZBX_SLA_PERIOD_QUARTERLY, ZBX_SLA_PERIOD_ANNUALLY])],
'slo' => ['type' => API_FLOAT, 'flags' => API_REQUIRED, 'in' => '0:100'],
'effective_date' => ['type' => API_INT32, 'flags' => API_REQUIRED, 'in' => '0:'.ZBX_MAX_DATE],
'timezone' => ['type' => API_STRING_UTF8, 'flags' => API_REQUIRED, 'in' => implode(',', array_merge([ZBX_DEFAULT_TIMEZONE], array_keys(CTimezoneHelper::getList()))), 'length' => DB::getFieldLength('sla', 'timezone')],
'status' => ['type' => API_INT32, 'in' => implode(',', [ZBX_SLA_STATUS_DISABLED, ZBX_SLA_STATUS_ENABLED])],
'description' => ['type' => API_STRING_UTF8, 'length' => DB::getFieldLength('sla', 'description')],
'service_tags' => ['type' => API_OBJECTS, 'flags' => API_REQUIRED | API_NOT_EMPTY, 'uniq' => [['tag', 'value']], 'fields' => [
'tag' => ['type' => API_STRING_UTF8, 'flags' => API_REQUIRED | API_NOT_EMPTY, 'length' => DB::getFieldLength('sla_service_tag', 'tag')],
'operator' => ['type' => API_INT32, 'in' => implode(',', [ZBX_SLA_SERVICE_TAG_OPERATOR_EQUAL, ZBX_SLA_SERVICE_TAG_OPERATOR_LIKE]), 'default' => DB::getDefault('sla_service_tag', 'operator')],
'value' => ['type' => API_STRING_UTF8, 'length' => DB::getFieldLength('sla_service_tag', 'value'), 'default' => DB::getDefault('sla_service_tag', 'value')]
]],
'schedule' => ['type' => API_OBJECTS, 'uniq' => [['period_from', 'period_to']], 'fields' => [
'period_from' => ['type' => API_INT32, 'flags' => API_REQUIRED, 'in' => '0:'.SEC_PER_WEEK],
'period_to' => ['type' => API_INT32, 'flags' => API_REQUIRED, 'in' => '0:'.SEC_PER_WEEK]
]],
'excluded_downtimes' => ['type' => API_OBJECTS, 'uniq' => [['period_from', 'period_to']], 'fields' => [
'name' => ['type' => API_STRING_UTF8, 'flags' => API_REQUIRED | API_NOT_EMPTY, 'length' => DB::getFieldLength('sla_excluded_downtime', 'name')],
'period_from' => ['type' => API_INT32, 'flags' => API_REQUIRED, 'in' => '0:'.ZBX_MAX_DATE],
'period_to' => ['type' => API_INT32, 'flags' => API_REQUIRED, 'in' => '0:'.ZBX_MAX_DATE]
]]
]];
if (!CApiInputValidator::validate($api_input_rules, $slas, '/', $error)) {
self::exception(ZBX_API_ERROR_PARAMETERS, $error);
}
self::checkDuplicates($slas);
self::checkSlo($slas);
self::checkSchedule($slas);
self::checkExcludedDowntimes($slas);
}
/**
* @param array $slas
*
* @throws APIException
*
* @return array
*/
public function update(array $slas): array {
self::validateUpdate($slas, $db_slas);
$upd_slas = [];
foreach ($slas as $sla) {
$upd_sla = DB::getUpdatedValues('sla', $sla, $db_slas[$sla['slaid']]);
if ($upd_sla) {
$upd_slas[] = [
'values' => $upd_sla,
'where' => ['slaid' => $sla['slaid']]
];
}
}
if ($upd_slas) {
DB::update('sla', $upd_slas);
}
self::updateServiceTags($slas, $db_slas);
self::updateSchedule($slas, $db_slas);
self::updateExcludedDowntimes($slas, $db_slas);
self::addAuditLog(CAudit::ACTION_UPDATE, CAudit::RESOURCE_SLA, $slas, $db_slas);
return ['slaids' => array_column($slas, 'slaid')];
}
/**
* @param array $slas
* @param array|null $db_slas
*
* @throws APIException
*/
private static function validateUpdate(array &$slas, ?array &$db_slas): void {
$api_input_rules = ['type' => API_OBJECTS, 'flags' => API_NOT_EMPTY | API_NORMALIZE, 'uniq' => [['slaid'], ['name']], 'fields' => [
'slaid' => ['type' => API_ID, 'flags' => API_REQUIRED],
'name' => ['type' => API_STRING_UTF8, 'flags' => API_NOT_EMPTY, 'length' => DB::getFieldLength('sla', 'name')],
'period' => ['type' => API_INT32, 'in' => implode(',', [ZBX_SLA_PERIOD_DAILY, ZBX_SLA_PERIOD_WEEKLY, ZBX_SLA_PERIOD_MONTHLY, ZBX_SLA_PERIOD_QUARTERLY, ZBX_SLA_PERIOD_ANNUALLY])],
'slo' => ['type' => API_FLOAT, 'in' => '0:100'],
'effective_date' => ['type' => API_INT32, 'in' => '0:'.ZBX_MAX_DATE],
'timezone' => ['type' => API_STRING_UTF8, 'in' => implode(',', array_merge([ZBX_DEFAULT_TIMEZONE], array_keys(CTimezoneHelper::getList()))), 'length' => DB::getFieldLength('sla', 'timezone')],
'status' => ['type' => API_INT32, 'in' => implode(',', [ZBX_SLA_STATUS_DISABLED, ZBX_SLA_STATUS_ENABLED])],
'description' => ['type' => API_STRING_UTF8, 'length' => DB::getFieldLength('sla', 'description')],
'service_tags' => ['type' => API_OBJECTS, 'flags' => API_NOT_EMPTY, 'uniq' => [['tag', 'value']], 'fields' => [
'tag' => ['type' => API_STRING_UTF8, 'flags' => API_REQUIRED | API_NOT_EMPTY, 'length' => DB::getFieldLength('sla_service_tag', 'tag')],
'operator' => ['type' => API_INT32, 'in' => implode(',', [ZBX_SLA_SERVICE_TAG_OPERATOR_EQUAL, ZBX_SLA_SERVICE_TAG_OPERATOR_LIKE]), 'default' => DB::getDefault('sla_service_tag', 'operator')],
'value' => ['type' => API_STRING_UTF8, 'length' => DB::getFieldLength('sla_service_tag', 'value'), 'default' => DB::getDefault('sla_service_tag', 'value')]
]],
'schedule' => ['type' => API_OBJECTS, 'uniq' => [['period_from', 'period_to']], 'fields' => [
'period_from' => ['type' => API_INT32, 'flags' => API_REQUIRED, 'in' => '0:'.SEC_PER_WEEK],
'period_to' => ['type' => API_INT32, 'flags' => API_REQUIRED, 'in' => '0:'.SEC_PER_WEEK]
]],
'excluded_downtimes' => ['type' => API_OBJECTS, 'uniq' => [['period_from', 'period_to']], 'fields' => [
'name' => ['type' => API_STRING_UTF8, 'flags' => API_REQUIRED | API_NOT_EMPTY, 'length' => DB::getFieldLength('sla_excluded_downtime', 'name')],
'period_from' => ['type' => API_INT32, 'flags' => API_REQUIRED, 'in' => '0:'.ZBX_MAX_DATE],
'period_to' => ['type' => API_INT32, 'flags' => API_REQUIRED, 'in' => '0:'.ZBX_MAX_DATE]
]]
]];
if (!CApiInputValidator::validate($api_input_rules, $slas, '/', $error)) {
self::exception(ZBX_API_ERROR_PARAMETERS, $error);
}
$db_slas = DB::select('sla', [
'output' => ['slaid', 'name', 'period', 'slo', 'timezone', 'status', 'description'],
'slaids' => array_column($slas, 'slaid'),
'preservekeys' => true
]);
if (count($db_slas) != count($slas)) {
self::exception(ZBX_API_ERROR_PERMISSIONS, _('No permissions to referred object or it does not exist!'));
}
self::addAffectedObjects($slas, $db_slas);
self::checkDuplicates($slas, $db_slas);
self::checkSlo($slas, $db_slas);
self::checkSchedule($slas, $db_slas);
self::checkExcludedDowntimes($slas, $db_slas);
}
/**
* @param array $slaids
*
* @throws APIException
*
* @return array
*/
public function delete(array $slaids): array {
$this->validateDelete($slaids, $db_slas);
DB::delete('sla', ['slaid' => $slaids]);
self::addAuditLog(CAudit::ACTION_DELETE, CAudit::RESOURCE_SLA, $db_slas);
return ['slaids' => $slaids];
}
/**
* @param array $slaids
* @param array|null $db_slas
*
* @throws APIException
*/
private function validateDelete(array $slaids, ?array &$db_slas): void {
$api_input_rules = ['type' => API_IDS, 'flags' => API_NOT_EMPTY, 'uniq' => true];
if (!CApiInputValidator::validate($api_input_rules, $slaids, '/', $error)) {
self::exception(ZBX_API_ERROR_PARAMETERS, $error);
}
$db_slas = $this->get([
'output' => ['slaid', 'name'],
'slaids' => $slaids,
'editable' => true,
'preservekeys' => true
]);
if (count($db_slas) != count($slaids)) {
self::exception(ZBX_API_ERROR_PERMISSIONS, _('No permissions to referred object or it does not exist!'));
}
}
/**
* @param string $table_name
* @param string $table_alias
* @param array $options
* @param array $sql_parts
*
* @return array
*/
protected function applyQueryFilterOptions($table_name, $table_alias, array $options, array $sql_parts): array {
$sql_parts = parent::applyQueryFilterOptions($table_name, $table_alias, $options, $sql_parts);
if ($options['service_tags']) {
$sql_parts['where'][] = CApiTagHelper::addWhereCondition($options['service_tags'], $options['evaltype'],
'sla', 'sla_service_tag', 'slaid'
);
}
return $sql_parts;
}
/**
* @param array $options
* @param array $result
*
* @return array
*/
protected function addRelatedObjects(array $options, array $result): array {
$result = parent::addRelatedObjects($options, $result);
self::addRelatedServiceTags($options, $result);
self::addRelatedSchedule($options, $result);
self::addRelatedExcludedDowntimes($options, $result);
return $result;
}
/**
* @param array $options
* @param array $result
*/
private static function addRelatedServiceTags(array $options, array &$result): void {
if ($options['selectServiceTags'] === null) {
return;
}
foreach ($result as &$row) {
$row['service_tags'] = [];
}
unset($row);
if ($options['selectServiceTags'] === API_OUTPUT_COUNT) {
$output = ['sla_service_tagid', 'slaid'];
}
elseif ($options['selectServiceTags'] === API_OUTPUT_EXTEND) {
$output = ['sla_service_tagid', 'slaid', 'tag', 'operator', 'value'];
}
else {
$output = array_unique(array_merge(['sla_service_tagid', 'slaid'], $options['selectServiceTags']));
}
$sql_options = [
'output' => $output,
'filter' => ['slaid' => array_keys($result)]
];
$db_service_tags = DBselect(DB::makeSql('sla_service_tag', $sql_options));
while ($db_service_tag = DBfetch($db_service_tags)) {
$slaid = $db_service_tag['slaid'];
unset($db_service_tag['sla_service_tagid'], $db_service_tag['slaid']);
$result[$slaid]['service_tags'][] = $db_service_tag;
}
if ($options['selectServiceTags'] === API_OUTPUT_COUNT) {
foreach ($result as &$row) {
$row['service_tags'] = (string) count($row['service_tags']);
}
unset($row);
}
}
/**
* @param array $options
* @param array $result
*/
private static function addRelatedSchedule(array $options, array &$result): void {
if ($options['selectSchedule'] === null) {
return;
}
foreach ($result as &$row) {
$row['schedule'] = [];
}
unset($row);
$sql_options = [
'output' => ['sla_scheduleid', 'slaid', 'period_from', 'period_to'],
'filter' => ['slaid' => array_keys($result)]
];
$db_schedule = DBselect(DB::makeSql('sla_schedule', $sql_options));
while ($db_schedule_row = DBfetch($db_schedule)) {
$slaid = $db_schedule_row['slaid'];
unset($db_schedule_row['sla_scheduleid'], $db_schedule_row['slaid']);
$result[$slaid]['schedule'][] = $db_schedule_row;
}
foreach ($result as &$row) {
$row['schedule'] = self::normalizeSchedule($row['schedule']);
if ($options['selectSchedule'] === API_OUTPUT_COUNT) {
$row['schedule'] = (string) count($row['schedule']);
}
elseif ($options['selectSchedule'] !== API_OUTPUT_EXTEND) {
$select_schedule = array_flip($options['selectSchedule']);
foreach ($row['schedule'] as &$schedule_row) {
$schedule_row = array_intersect_key($schedule_row, $select_schedule);
}
unset($schedule_row);
}
}
unset($row);
}
/**
* @param array $options
* @param array $result
*/
private static function addRelatedExcludedDowntimes(array $options, array &$result): void {
if ($options['selectExcludedDowntimes'] === null) {
return;
}
foreach ($result as &$row) {
$row['excluded_downtimes'] = [];
}
unset($row);
if ($options['selectExcludedDowntimes'] === API_OUTPUT_COUNT) {
$output = ['sla_excluded_downtimeid', 'slaid'];
}
elseif ($options['selectExcludedDowntimes'] === API_OUTPUT_EXTEND) {
$output = ['sla_excluded_downtimeid', 'slaid', 'name', 'period_from', 'period_to'];
}
else {
$output = array_unique(array_merge(['sla_excluded_downtimeid', 'slaid'],
$options['selectExcludedDowntimes']
));
}
$sql_options = [
'output' => $output,
'filter' => ['slaid' => array_keys($result)],
'sortfield' => ['period_from'],
'sortorder' => [ZBX_SORT_UP]
];
$db_sla_excluded_downtimes = DBselect(DB::makeSql('sla_excluded_downtime', $sql_options));
while ($db_sla_excluded_downtime = DBfetch($db_sla_excluded_downtimes)) {
$slaid = $db_sla_excluded_downtime['slaid'];
unset($db_sla_excluded_downtime['sla_excluded_downtimeid'], $db_sla_excluded_downtime['slaid']);
$result[$slaid]['excluded_downtimes'][] = $db_sla_excluded_downtime;
}
if ($options['selectExcludedDowntimes'] === API_OUTPUT_COUNT) {
foreach ($result as &$row) {
$row['excluded_downtimes'] = (string) count($row['excluded_downtimes']);
}
unset($row);
}
}
/**
* @param array $slas
* @param array|null $db_slas
*
* @throws APIException
*/
private static function checkDuplicates(array $slas, array $db_slas = null): void {
$names = [];
foreach ($slas as $sla) {
if ($db_slas === null
|| (array_key_exists('name', $sla) && $sla['name'] !== $db_slas[$sla['slaid']]['name'])) {
$names[] = $sla['name'];
}
}
if (!$names) {
return;
}
$duplicate = DBfetch(DBselect(
'SELECT sla.name FROM sla WHERE '.dbConditionString('sla.name', $names), 1
));
if ($duplicate) {
self::exception(ZBX_API_ERROR_PARAMETERS, _s('SLA "%1$s" already exists.', $duplicate['name']));
}
}
/**
* @param array $slas
* @param array|null $db_slas
*
* @throws APIException
*/
private static function checkSlo(array $slas, array $db_slas = null): void {
foreach ($slas as $sla) {
$name = $db_slas !== null ? $db_slas[$sla['slaid']]['name'] : $sla['name'];
if (array_key_exists('slo', $sla) && round($sla['slo'], 4) != $sla['slo']) {
self::exception(ZBX_API_ERROR_PARAMETERS,
_s('SLA "%1$s" SLO must have no more than 4 fractional digits.', $name)
);
}
}
}
/**
* @param array $slas
* @param array|null $db_slas
*
* @throws APIException
*/
private static function checkSchedule(array &$slas, array $db_slas = null): void {
foreach ($slas as &$sla) {
if (!array_key_exists('schedule', $sla)) {
continue;
}
$name = $db_slas !== null ? $db_slas[$sla['slaid']]['name'] : $sla['name'];
foreach ($sla['schedule'] as $schedule_row) {
if ($schedule_row['period_from'] >= $schedule_row['period_to']) {
self::exception(ZBX_API_ERROR_PARAMETERS,
_s('Start time must be less than end time for SLA "%1$s".', $name)
);
}
}
$sla['schedule'] = self::normalizeSchedule($sla['schedule']);
}
unset($sla);
}
/**
* @param array $slas
* @param array|null $db_slas
*
* @throws APIException
*/
private static function checkExcludedDowntimes(array $slas, array $db_slas = null): void {
foreach ($slas as $sla) {
if (!array_key_exists('excluded_downtimes', $sla)) {
continue;
}
$name = $db_slas !== null ? $db_slas[$sla['slaid']]['name'] : $sla['name'];
foreach ($sla['excluded_downtimes'] as $excluded_downtime) {
if ($excluded_downtime['period_from'] >= $excluded_downtime['period_to']) {
self::exception(ZBX_API_ERROR_PARAMETERS, _s(
'Start time must be less than end time for excluded downtime "%2$s" of SLA "%1$s".',
$name, $excluded_downtime['name']
));
}
}
}
}
/**
* @param array $slas
* @param array $db_slas
*/
private static function addAffectedObjects(array $slas, array &$db_slas): void {
self::addAffectedServiceTags($slas, $db_slas);
self::addAffectedSchedule($slas, $db_slas);
self::addAffectedExcludedDowntimes($slas, $db_slas);
}
/**
* @param array $slas
* @param array $db_slas
*/
private static function addAffectedServiceTags(array $slas, array &$db_slas): void {
$affected_slaids = [];
foreach ($slas as $sla) {
if (array_key_exists('service_tags', $sla)) {
$affected_slaids[$sla['slaid']] = true;
$db_slas[$sla['slaid']]['service_tags'] = [];
}
}
$sql_options = [
'output' => ['sla_service_tagid', 'slaid', 'tag', 'operator', 'value'],
'filter' => ['slaid' => array_keys($affected_slaids)]
];
$db_service_tags = DBselect(DB::makeSql('sla_service_tag', $sql_options));
while ($db_service_tag = DBfetch($db_service_tags)) {
$slaid = $db_service_tag['slaid'];
unset($db_service_tag['slaid']);
$db_slas[$slaid]['service_tags'][$db_service_tag['sla_service_tagid']] = $db_service_tag;
}
}
/**
* @param array $slas
* @param array $db_slas
*/
private static function addAffectedSchedule(array $slas, array &$db_slas): void {
$affected_slaids = [];
foreach ($slas as $sla) {
if (array_key_exists('schedule', $sla)) {
$affected_slaids[$sla['slaid']] = true;
$db_slas[$sla['slaid']]['schedule'] = [];
}
}
$sql_options = [
'output' => ['sla_scheduleid', 'slaid', 'period_from', 'period_to'],
'filter' => ['slaid' => array_keys($affected_slaids)]
];
$db_schedule = DBselect(DB::makeSql('sla_schedule', $sql_options));
while ($db_schedule_row = DBfetch($db_schedule)) {
$slaid = $db_schedule_row['slaid'];
unset($db_schedule_row['slaid']);
$db_slas[$slaid]['schedule'][$db_schedule_row['sla_scheduleid']] = $db_schedule_row;
}
}
/**
* @param array $slas
* @param array $db_slas
*/
private static function addAffectedExcludedDowntimes(array $slas, array &$db_slas): void {
$affected_slaids = [];
foreach ($slas as $sla) {
if (array_key_exists('excluded_downtimes', $sla)) {
$affected_slaids[$sla['slaid']] = true;
$db_slas[$sla['slaid']]['excluded_downtimes'] = [];
}
}
$sql_options = [
'output' => ['sla_excluded_downtimeid', 'slaid', 'name', 'period_from', 'period_to'],
'filter' => ['slaid' => array_keys($affected_slaids)]
];
$db_excluded_downtimes = DBselect(DB::makeSql('sla_excluded_downtime', $sql_options));
while ($db_excluded_downtime = DBfetch($db_excluded_downtimes)) {
$slaid = $db_excluded_downtime['slaid'];
unset($db_excluded_downtime['slaid']);
$db_slas[$slaid]['excluded_downtimes'][$db_excluded_downtime['sla_excluded_downtimeid']] =
$db_excluded_downtime;
}
}
/**
* @param array $slas
* @param array|null $db_slas
*/
private static function updateServiceTags(array &$slas, array $db_slas = null): void {
$ins_service_tags = [];
$del_service_tags = [];
foreach ($slas as &$sla) {
if (!array_key_exists('service_tags', $sla)) {
continue;
}
$db_service_tags = [];
if ($db_slas !== null) {
foreach ($db_slas[$sla['slaid']]['service_tags'] as $db_service_tag) {
$db_service_tags[$db_service_tag['tag']][$db_service_tag['operator']][$db_service_tag['value']] =
$db_service_tag['sla_service_tagid'];
$del_service_tags[$db_service_tag['sla_service_tagid']] = true;
}
}
foreach ($sla['service_tags'] as &$service_tag) {
if (array_key_exists($service_tag['tag'], $db_service_tags)
&& array_key_exists($service_tag['operator'], $db_service_tags[$service_tag['tag']])
&& array_key_exists($service_tag['value'],
$db_service_tags[$service_tag['tag']][$service_tag['operator']]
)) {
$service_tag['sla_service_tagid'] =
$db_service_tags[$service_tag['tag']][$service_tag['operator']][$service_tag['value']];
unset($del_service_tags[$service_tag['sla_service_tagid']]);
}
else {
$ins_service_tags[] = ['slaid' => $sla['slaid']] + $service_tag;
}
}
unset($service_tag);
}
unset($sla);
if ($del_service_tags) {
DB::delete('sla_service_tag', ['sla_service_tagid' => array_keys($del_service_tags)]);
}
if ($ins_service_tags) {
$sla_service_tagids = DB::insert('sla_service_tag', $ins_service_tags);
$sla_service_tagids_index = 0;
foreach ($slas as &$sla) {
if (!array_key_exists('service_tags', $sla)) {
continue;
}
foreach ($sla['service_tags'] as &$service_tag) {
if (array_key_exists('sla_service_tagid', $service_tag)) {
continue;
}
$service_tag['sla_service_tagid'] = $sla_service_tagids[$sla_service_tagids_index];
$sla_service_tagids_index++;
}
unset($service_tag);
}
unset($sla);
}
}
/**
* @param array $slas
* @param array|null $db_slas
*/
private static function updateSchedule(array &$slas, array $db_slas = null): void {
$ins_schedule = [];
$del_schedule = [];
foreach ($slas as &$sla) {
if (!array_key_exists('schedule', $sla)) {
continue;
}
$db_schedule = [];
if ($db_slas !== null) {
foreach ($db_slas[$sla['slaid']]['schedule'] as $db_schedule_row) {
$db_schedule[$db_schedule_row['period_from']][$db_schedule_row['period_to']] =
$db_schedule_row['sla_scheduleid'];
$del_schedule[$db_schedule_row['sla_scheduleid']] = true;
}
}
foreach ($sla['schedule'] as &$schedule_row) {
if (array_key_exists($schedule_row['period_from'], $db_schedule)
&& array_key_exists($schedule_row['period_to'], $db_schedule[$schedule_row['period_from']])) {
$schedule_row['sla_scheduleid'] =
$db_schedule[$schedule_row['period_from']][$schedule_row['period_to']];
unset($del_schedule[$schedule_row['sla_scheduleid']]);
}
else {
$ins_schedule[] = ['slaid' => $sla['slaid']] + $schedule_row;
}
}
unset($schedule_row);
}
unset($sla);
if ($del_schedule) {
DB::delete('sla_schedule', ['sla_scheduleid' => array_keys($del_schedule)]);
}
if ($ins_schedule) {
$sla_scheduleids = DB::insert('sla_schedule', $ins_schedule);
$sla_scheduleids_index = 0;
foreach ($slas as &$sla) {
if (!array_key_exists('schedule', $sla)) {
continue;
}
foreach ($sla['schedule'] as &$schedule_row) {
if (array_key_exists('sla_scheduleid', $schedule_row)) {
continue;
}
$schedule_row['sla_scheduleid'] = $sla_scheduleids[$sla_scheduleids_index];
$sla_scheduleids_index++;
}
unset($schedule_row);
}
unset($sla);
}
}
/**
* @param array $slas
* @param array|null $db_slas
*/
private static function updateExcludedDowntimes(array &$slas, array $db_slas = null): void {
$ins_excluded_downtimes = [];
$upd_excluded_downtimes = [];
$del_excluded_downtimes = [];
foreach ($slas as &$sla) {
if (!array_key_exists('excluded_downtimes', $sla)) {
continue;
}
$db_excluded_downtimes = [];
if ($db_slas !== null) {
foreach ($db_slas[$sla['slaid']]['excluded_downtimes'] as $db_excluded_downtime) {
$db_excluded_downtimes[$db_excluded_downtime['period_from']][$db_excluded_downtime['period_to']] =
$db_excluded_downtime;
$del_excluded_downtimes[$db_excluded_downtime['sla_excluded_downtimeid']] = true;
}
}
foreach ($sla['excluded_downtimes'] as &$excluded_downtime) {
if (array_key_exists($excluded_downtime['period_from'], $db_excluded_downtimes)
&& array_key_exists($excluded_downtime['period_to'],
$db_excluded_downtimes[$excluded_downtime['period_from']]
)) {
$excluded_downtime['sla_excluded_downtimeid'] = $db_excluded_downtimes
[$excluded_downtime['period_from']][$excluded_downtime['period_to']]['sla_excluded_downtimeid'];
unset($del_excluded_downtimes[$excluded_downtime['sla_excluded_downtimeid']]);
$upd_excluded_downtime = DB::getUpdatedValues('sla_excluded_downtime', $excluded_downtime,
$db_excluded_downtimes[$excluded_downtime['period_from']][$excluded_downtime['period_to']]
);
if ($upd_excluded_downtime) {
$upd_excluded_downtimes[] = [
'values' => $upd_excluded_downtime,
'where' => ['sla_excluded_downtimeid' => $excluded_downtime['sla_excluded_downtimeid']]
];
}
}
else {
$ins_excluded_downtimes[] = ['slaid' => $sla['slaid']] + $excluded_downtime;
}
}
unset($excluded_downtime);
}
unset($sla);
if ($del_excluded_downtimes) {
DB::delete('sla_excluded_downtime', ['sla_excluded_downtimeid' => array_keys($del_excluded_downtimes)]);
}
if ($ins_excluded_downtimes) {
$sla_excluded_downtimeids = DB::insert('sla_excluded_downtime', $ins_excluded_downtimes);
$sla_excluded_downtimeids_index = 0;
foreach ($slas as &$sla) {
if (!array_key_exists('excluded_downtimes', $sla)) {
continue;
}
foreach ($sla['excluded_downtimes'] as &$excluded_downtime) {
if (array_key_exists('sla_excluded_downtimeid', $excluded_downtime)) {
continue;
}
$excluded_downtime['sla_excluded_downtimeid'] =
$sla_excluded_downtimeids[$sla_excluded_downtimeids_index];
$sla_excluded_downtimeids_index++;
}
unset($excluded_downtime);
}
unset($sla);
}
if ($upd_excluded_downtimes) {
DB::update('sla_excluded_downtime', $upd_excluded_downtimes);
}
}
/**
* Concatenate overlapping periods and sort by starting time of periods.
* Full weekly period is returned as empty array.
*
* @param array $schedule
*
* @return array
*/
private static function normalizeSchedule(array $schedule): array {
$converted_schedule = [];
foreach ($schedule as $schedule_row) {
$period_from = $schedule_row['period_from'];
$period_to = $schedule_row['period_to'];
foreach ($converted_schedule as $converted_schedule_row) {
$is_overlapping = $schedule_row['period_from'] <= $converted_schedule_row['period_to']
&& $schedule_row['period_to'] >= $converted_schedule_row['period_from'];
if ($is_overlapping) {
$period_from = min($period_from, $converted_schedule_row['period_from']);
$period_to = max($period_to, $converted_schedule_row['period_to']);
}
}
foreach ($converted_schedule as $index => $converted_schedule_row) {
if ($converted_schedule_row['period_from'] >= $period_from
&& $converted_schedule_row['period_to'] <= $period_to) {
unset($converted_schedule[$index]);
}
}
$converted_schedule[] = ['period_from' => $period_from, 'period_to' => $period_to];
}
usort($converted_schedule,
static function (array $schedule_row_a, array $schedule_row_b): int {
return $schedule_row_a['period_from'] <=> $schedule_row_b['period_from'];
}
);
if (count($converted_schedule) == 1 && $converted_schedule[0]['period_from'] == 0
&& $converted_schedule[0]['period_to'] == SEC_PER_WEEK) {
return [];
}
return $converted_schedule;
}
/**
* @param array|null $limit_slaids
* @param array|null $limit_serviceids
*
* @throws APIException
*
* @return array|null
*/
private static function getAccessibleSlaids(?array $limit_slaids, ?array $limit_serviceids): ?array {
$role = API::Role()->get([
'output' => [],
'selectRules' => ['services.read.mode', 'services.write.mode', 'actions'],
'roleids' => self::$userData['roleid']
]);
if (!$role) {
return [];
}
if ($limit_serviceids === null) {
$rules = $role[0]['rules'];
$manage_sla_status = 0;
foreach ($rules['actions'] as $action) {
if ($action['name'] === 'manage_sla') {
$manage_sla_status = $action['status'];
break;
}
}
if ($rules['services.read.mode'] == ZBX_ROLE_RULE_SERVICES_ACCESS_ALL
|| $rules['services.write.mode'] == ZBX_ROLE_RULE_SERVICES_ACCESS_ALL
|| $manage_sla_status == 1) {
return null;
}
}
$accessible_services = API::Service()->get([
'output' => [],
'serviceids' => $limit_serviceids,
'preservekeys' => true
]);
if (!$accessible_services) {
return [];
}
$services_slas_resource_sql = 'SELECT DISTINCT st.serviceid,sst.slaid'.
' FROM service_tag st, sla_service_tag sst'.
' WHERE sst.tag=st.tag'.
' AND ('.
'(sst.operator='.ZBX_SLA_SERVICE_TAG_OPERATOR_EQUAL.' AND st.value=sst.value)'.
' OR (sst.operator='.ZBX_SLA_SERVICE_TAG_OPERATOR_LIKE." AND UPPER(st.value) LIKE CONCAT('%', ".
"CONCAT(REPLACE(REPLACE(REPLACE(UPPER(sst.value), '%', '!%'), '_', '!_'), '!', '!!'), '%')".
") ESCAPE '!')".
')'.
($limit_slaids !== null ? ' AND '.dbConditionId('sst.slaid', $limit_slaids) : '').
($limit_serviceids !== null ? ' AND '.dbConditionId('st.serviceid', $limit_serviceids) : '');
$services_slas_resource = DBSelect($services_slas_resource_sql);
$accessible_slaids = [];
while ($db_service_sla = DBfetch($services_slas_resource)) {
if (array_key_exists($db_service_sla['serviceid'], $accessible_services)) {
$accessible_slaids[$db_service_sla['slaid']] = true;
}
}
return $accessible_slaids;
}
/**
* @param array $options
*
* @throws Exception
* @throws APIException
*
* @return array
*/
public function getSli(array $options = []): array {
$api_input_rules = ['type' => API_OBJECT, 'fields' => [
'slaid' => ['type' => API_ID, 'flags' => API_REQUIRED],
'period_from' => ['type' => API_INT32, 'flags' => API_ALLOW_NULL, 'in' => '0:'.ZBX_MAX_DATE, 'default' => null],
'period_to' => ['type' => API_INT32, 'flags' => API_ALLOW_NULL, 'in' => '0:'.ZBX_MAX_DATE, 'default' => null],
'periods' => ['type' => API_INT32, 'flags' => API_ALLOW_NULL, 'in' => '1:'.ZBX_SLA_MAX_REPORTING_PERIODS, 'default' => null],
'serviceids' => ['type' => API_IDS, 'flags' => API_ALLOW_NULL | API_NORMALIZE, 'default' => null]
]];
if (!CApiInputValidator::validate($api_input_rules, $options, '/', $error)) {
self::exception(ZBX_API_ERROR_PARAMETERS, $error);
}
$db_sla = $this->get([
'output' => ['period', 'slo', 'timezone', 'effective_date'],
'selectSchedule' => ['period_from', 'period_to'],
'selectExcludedDowntimes' => ['name', 'period_from', 'period_to'],
'slaids' => $options['slaid']
]);
if (!$db_sla) {
self::exception(ZBX_API_ERROR_PERMISSIONS, _('No permissions to referred object or it does not exist!'));
}
$db_sla = $db_sla[0];
$reporting_periods = self::getReportingPeriods($db_sla, $options);
if ($reporting_periods) {
$db_services = API::Service()->get([
'output' => ['created_at'],
'selectStatusTimeline' => $reporting_periods,
'slaids' => $options['slaid'],
'serviceids' => $options['serviceids'],
'preservekeys' => true
]);
}
else {
$db_services = [];
}
return [
'periods' => $reporting_periods,
'serviceids' => array_keys($db_services),
'sli' => self::calculateSli($db_sla, $reporting_periods, array_values($db_services))
];
}
/**
* @param array $sla
* @param array $options
*
* @throws Exception
*
* @return array
*/
private static function getReportingPeriods(array $sla, array $options): array {
$interval = new DateInterval([
ZBX_SLA_PERIOD_DAILY => 'P1D',
ZBX_SLA_PERIOD_WEEKLY => 'P1W',
ZBX_SLA_PERIOD_MONTHLY => 'P1M',
ZBX_SLA_PERIOD_QUARTERLY => 'P3M',
ZBX_SLA_PERIOD_ANNUALLY => 'P1Y'
][$sla['period']]);
$timezone = new DateTimeZone($sla['timezone'] !== ZBX_DEFAULT_TIMEZONE
? $sla['timezone']
: CTimezoneHelper::getSystemTimezone()
);
$effective_min = (new DateTime('@'.$sla['effective_date']))->setTimezone($timezone);
self::alignDateToPeriodStart($effective_min, (int) $sla['period']);
$effective_max = (new DateTime('now'))->setTimezone($timezone);
self::alignDateToPeriodStart($effective_max, (int) $sla['period']);
$effective_max->add($interval);
if ($options['period_from'] !== null) {
$period_from = (new DateTime('@'.$options['period_from']))->setTimezone($timezone);
self::alignDateToPeriodStart($period_from, (int) $sla['period']);
if ($period_from->getTimestamp() < 0) {
$period_from->add($interval);
}
}
else {
$period_from = null;
}
if ($options['period_to'] !== null) {
$period_to = (new DateTime('@'.$options['period_to']))->setTimezone($timezone);
self::alignDateToPeriodStart($period_to, (int) $sla['period']);
$period_to->add($interval);
if ($period_to->getTimestamp() > ZBX_MAX_DATE) {
$period_to->sub($interval);
}
}
elseif ($period_from === null) {
$period_to = $effective_max;
}
else {
$period_to = null;
}
$reporting_periods = [];
$do_descend = $period_to !== null;
$date = $do_descend ? clone $period_to : clone $period_from;
while (count($reporting_periods) < ZBX_SLA_MAX_REPORTING_PERIODS) {
if ($options['periods'] !== null) {
if (count($reporting_periods) == $options['periods']) {
break;
}
}
else {
if (($period_from === null || $period_to === null)
&& count($reporting_periods) == ZBX_SLA_DEFAULT_REPORTING_PERIODS) {
break;
}
if ($do_descend) {
if ($period_from === null && $date <= $effective_min) {
break;
}
}
elseif ($date >= $effective_max) {
break;
}
}
if ($do_descend && $period_from !== null && $date <= $period_from) {
break;
}
if ($do_descend) {
$to = $date->getTimestamp();
$date->sub($interval);
$from = $date->getTimestamp();
if ($from < 0) {
break;
}
array_unshift($reporting_periods, ['period_from' => $from, 'period_to' => $to]);
}
else {
$from = $date->getTimestamp();
$date->add($interval);
$to = $date->getTimestamp();
if ($to > ZBX_MAX_DATE) {
break;
}
$reporting_periods[] = ['period_from' => $from, 'period_to' => $to];
}
}
return $reporting_periods;
}
/**
* @param DateTime $date
*
* @param int $sla_period
*/
private static function alignDateToPeriodStart(DateTime $date, int $sla_period): void {
$year = (int) $date->format('Y');
$month = (int) $date->format('n');
switch ($sla_period) {
case ZBX_SLA_PERIOD_WEEKLY:
$date
->modify('1 day')
->modify('last Sunday');
break;
case ZBX_SLA_PERIOD_MONTHLY:
$date->setDate($year, $month, 1);
break;
case ZBX_SLA_PERIOD_QUARTERLY:
$date->setDate($year, intdiv($month - 1, 3) * 3 + 1, 1);
break;
case ZBX_SLA_PERIOD_ANNUALLY:
$date->setDate($year, 1, 1);
break;
}
$date->setTime(0, 0);
}
/**
* @param array $db_sla
* @param array $reporting_periods
* @param array $db_services
*
* @throws Exception
*
* @return array
*/
private static function calculateSli(array $db_sla, array $reporting_periods, array $db_services): array {
if (!$reporting_periods || !$db_services) {
return [];
}
$sli = [];
$combined_excluded_downtimes = self::combineExcludedDowntimes($db_sla['excluded_downtimes']);
foreach ($reporting_periods as $reporting_period_index => $reporting_period) {
$scheduled_uptime_periods = self::getScheduledUptimePeriods($db_sla, $reporting_period);
foreach ($db_services as $service_index => $db_service) {
$cell = [
'uptime' => 0,
'downtime' => 0,
'sli' => -1.0,
'error_budget' => 0,
'excluded_downtimes' => []
];
$max_uptime = 0;
foreach ($scheduled_uptime_periods as $scheduled_uptime_period) {
$uptime_period_from = max($db_service['created_at'], $scheduled_uptime_period['period_from']);
$uptime_period_to = $scheduled_uptime_period['period_to'];
$uptime = $uptime_period_to - $uptime_period_from;
if ($uptime <= 0) {
continue;
}
$max_uptime += $uptime;
foreach ($combined_excluded_downtimes as $combined_excluded_downtime) {
$downtime = min($combined_excluded_downtime['period_to'], $uptime_period_to)
- max($combined_excluded_downtime['period_from'], $uptime_period_from);
if ($downtime > 0) {
$max_uptime -= $downtime;
}
}
}
$last_excluded_downtimes = [];
$prev_clock = $reporting_period['period_from'];
$prev_value = $db_service['status_timeline'][$reporting_period_index]['start_value'];
$alarms = $db_service['status_timeline'][$reporting_period_index]['alarms'];
if (!$alarms || $alarms[count($alarms) - 1]['clock'] <= time()) {
$alarms[] = ['clock' => time() + 1, 'value' => null];
}
foreach ($alarms as $alarm) {
foreach ($scheduled_uptime_periods as $scheduled_uptime_period) {
$uptime_period_from = max($db_service['created_at'], $scheduled_uptime_period['period_from'],
$prev_clock
);
$uptime_period_to = min($scheduled_uptime_period['period_to'], $alarm['clock']);
$uptime = $uptime_period_to - $uptime_period_from;
if ($uptime <= 0) {
continue;
}
foreach ($combined_excluded_downtimes as $combined_excluded_downtime) {
$downtime = min($combined_excluded_downtime['period_to'], $uptime_period_to)
- max($combined_excluded_downtime['period_from'], $uptime_period_from);
if ($downtime > 0) {
$uptime -= $downtime;
}
}
if ($prev_value == ZBX_SEVERITY_OK) {
$cell['uptime'] += $uptime;
}
else {
$cell['downtime'] += $uptime;
}
foreach ($db_sla['excluded_downtimes'] as $index => $excluded_downtime) {
$downtime_period_from = max($excluded_downtime['period_from'], $uptime_period_from);
$downtime_period_to = min($excluded_downtime['period_to'], $uptime_period_to);
if ($downtime_period_to > $downtime_period_from) {
if (array_key_exists($index, $last_excluded_downtimes)) {
$cell['excluded_downtimes'][$last_excluded_downtimes[$index]['cell']]['period_to'] =
(int) $downtime_period_to;
}
else {
$cell['excluded_downtimes'][] = [
'name' => $excluded_downtime['name'],
'period_from' => (int) $downtime_period_from,
'period_to' => (int) $downtime_period_to
];
}
$last_excluded_downtimes[$index] = [
'cell' => count($cell['excluded_downtimes']) - 1,
'period_to' => $downtime_period_to
];
}
else {
unset($last_excluded_downtimes[$index]);
}
}
}
$prev_clock = $alarm['clock'];
$prev_value = $alarm['value'];
}
if ($cell['uptime'] + $cell['downtime'] != 0) {
$cell['sli'] = $cell['uptime'] / ($cell['uptime'] + $cell['downtime']) * 100;
}
if ($cell['sli'] != -1) {
$available_uptime = $max_uptime - $cell['uptime'] - $cell['downtime'];
$cell['error_budget'] = $db_sla['slo'] > 0
? min($available_uptime,
(int) ($cell['uptime'] / $db_sla['slo'] * 100) - $cell['uptime'] - $cell['downtime']
)
: $available_uptime;
}
$sli[$reporting_period_index][$service_index] = $cell;
}
}
return $sli;
}
/**
* @param array $excluded_downtimes
*
* @return array
*/
private static function combineExcludedDowntimes(array $excluded_downtimes): array {
$combined_excluded_downtimes = [];
foreach ($excluded_downtimes as $excluded_downtime) {
$period_from = $excluded_downtime['period_from'];
$period_to = $excluded_downtime['period_to'];
foreach ($combined_excluded_downtimes as $combined_excluded_downtime) {
$is_overlapping = $excluded_downtime['period_from'] <= $combined_excluded_downtime['period_to']
&& $excluded_downtime['period_to'] >= $combined_excluded_downtime['period_from'];
if ($is_overlapping) {
$period_from = min($period_from, $combined_excluded_downtime['period_from']);
$period_to = max($period_to, $combined_excluded_downtime['period_to']);
}
}
foreach ($combined_excluded_downtimes as $index => $combined_excluded_downtime) {
if ($combined_excluded_downtime['period_from'] >= $period_from
&& $combined_excluded_downtime['period_to'] <= $period_to) {
unset($combined_excluded_downtimes[$index]);
}
}
$combined_excluded_downtimes[] = ['period_from' => $period_from, 'period_to' => $period_to];
}
return $combined_excluded_downtimes;
}
/**
* @param array $db_sla
* @param array $reporting_period
*
* @throws Exception
*
* @return array
*/
private static function getScheduledUptimePeriods(array $db_sla, array $reporting_period): array {
if (!$db_sla['schedule']) {
return [$reporting_period];
}
$uptime_periods = [];
$reporting_period_from =
(new DateTimeImmutable('@'.$reporting_period['period_from']))
->setTimezone(new DateTimeZone($db_sla['timezone'] !== ZBX_DEFAULT_TIMEZONE
? $db_sla['timezone']
: CTimezoneHelper::getSystemTimezone()
))
->modify('1 day')
->modify('last Sunday');
for ($week = 0;; $week++) {
$week_period_from = $reporting_period_from->modify($week.' week');
foreach ($db_sla['schedule'] as $schedule_row) {
$period_from = $week_period_from
->modify((int) ($schedule_row['period_from'] / SEC_PER_DAY).' day')
->setTime(
(int) ($schedule_row['period_from'] / SEC_PER_HOUR) % 24,
(int) ($schedule_row['period_from'] / SEC_PER_MIN) % 60,
$schedule_row['period_from'] % 60
)
->getTimestamp();
$period_to = $week_period_from
->modify((int) ($schedule_row['period_to'] / SEC_PER_DAY).' day')
->setTime(
(int) ($schedule_row['period_to'] / SEC_PER_HOUR) % 24,
(int) ($schedule_row['period_to'] / SEC_PER_MIN) % 60,
$schedule_row['period_to'] % 60
)
->getTimestamp();
if ($period_from < $reporting_period['period_to'] && $period_to > $reporting_period['period_from']) {
$new_period_from = max($reporting_period['period_from'], $period_from);
$new_period_to = min($reporting_period['period_to'], $period_to);
if ($uptime_periods
&& $uptime_periods[count($uptime_periods) - 1]['period_to'] == $new_period_from) {
$uptime_periods[count($uptime_periods) - 1]['period_to'] = $new_period_to;
}
else {
$uptime_periods[] = ['period_from' => $new_period_from, 'period_to' => $new_period_to];
}
}
if ($period_to >= $reporting_period['period_to']) {
break 2;
}
}
}
return $uptime_periods;
}
}