#10194 sections Alerts and Events

This commit is contained in:
Daniel Cebrian 2023-09-27 13:30:56 +02:00
parent 0fdbefa7ee
commit 62d501e6d5
16 changed files with 839 additions and 26 deletions

View File

@ -0,0 +1,78 @@
<?php
/**
* Ajax secondary controller for general tactival view.
*
* @category Ajax general tactical view page.
* @package Pandora FMS
* @subpackage Opensource
* @version 1.0.0
* @license See below
*
* ______ ___ _______ _______ ________
* | __ \.-----.--.--.--| |.-----.----.-----. | ___| | | __|
* | __/| _ | | _ || _ | _| _ | | ___| |__ |
* |___| |___._|__|__|_____||_____|__| |___._| |___| |__|_|__|_______|
*
* ============================================================================
* Copyright (c) 2005-2023 Pandora FMS
* Please see https://pandorafms.com/community/ for full contribution list
* 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 for version 2.
* 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.
* ============================================================================
*/
// Begin.
global $config;
// Only logged users have access to this endpoint.
check_login();
if (! check_acl($config['id_user'], 0, 'AR')) {
db_pandora_audit(
AUDIT_LOG_ACL_VIOLATION,
'Trying to access credential store'
);
if (is_ajax()) {
echo json_encode(['error' => 'noaccess']);
} else {
include 'general/noaccess.php';
}
exit;
}
// AJAX controller.
if (is_ajax()) {
$dir = $config['homedir'].'/include/lib/TacticalView/elements/';
$method = get_parameter('method');
$class = get_parameter('class');
$filepath = realpath($dir.'/'.$class.'.php');
if (is_readable($filepath) === false
|| is_dir($filepath) === true
|| preg_match('/.*\.php$/', $filepath) === false
) {
exit;
}
include_once $filepath;
if (class_exists($class) === true) {
$instance = new $class();
if ($instance->ajaxMethod($method) === true) {
$instance->{$method}();
} else {
$instance->error('Unavailable method.');
}
} else {
$class->error('Class not found. ['.$class.']');
}
exit;
}

View File

@ -2764,6 +2764,7 @@ function graph_agent_status(
'height' => $height,
'colors' => array_values($colors),
'legend' => ['display' => false],
'labels' => array_keys($data),
];
if ($donut_narrow_graph == true) {

View File

@ -1216,6 +1216,13 @@ function get_build_setup_charts($type, $options, $data)
$borders = array_values($defaultBorder);
}
if (isset($options['borderColors']) === true
&& empty($options['borderColors']) === false
&& is_array($options['borderColors']) === true
) {
$borders = $options['borderColors'];
}
// Set labels.
if (isset($options['labels']) === true
&& empty($options['labels']) === false

View File

@ -28,6 +28,20 @@ namespace PandoraFMS\TacticalView;
class Element
{
/**
* Url of controller.
*
* @var string
*/
public $ajaxController;
/**
* List of available ajax methods.
*
* @var array
*/
protected $ajaxMethods = [];
/**
* Title of section
*
@ -44,12 +58,42 @@ class Element
/**
* Constructor
* Contructor
*
* @param string $ajax_controller Controller.
*/
public function __construct()
{
public function __construct(
$ajax_controller='include/ajax/general_tactical_view.ajax'
) {
$this->interval = 0;
$this->title = __('Default element');
$this->ajaxController = $ajax_controller;
}
/**
* Return error message to target.
*
* @param string $msg Error message.
*
* @return void
*/
public static function error(string $msg)
{
echo json_encode(['error' => $msg]);
}
/**
* Verifies target method is allowed to be called using AJAX call.
*
* @param string $method Method to be invoked via AJAX.
*
* @return boolean Available (true), or not (false).
*/
public function ajaxMethod(string $method):bool
{
return in_array($method, $this->ajaxMethods) === true;
}

View File

@ -121,9 +121,9 @@ class GeneralTacticalView
private function getWelcomeMessage():string
{
global $config;
$profile = users_get_user_profile($config['id_user']);
if (is_array($profile) === true && count($profile) > 0) {
$name = $profile[0]['name'];
$user = users_get_user_by_id($config['id_user']);
if (is_array($user) === true && count($user) > 0) {
$name = $user['fullname'];
} else {
$name = '';
}

View File

@ -0,0 +1,220 @@
<?php
/**
* Alerts element for tactical view.
*
* @category General
* @package Pandora FMS
* @subpackage TacticalView
* @version 1.0.0
* @license See below
*
* ______ ___ _______ _______ ________
* | __ \.-----.--.--.--| |.-----.----.-----. | ___| | | __|
* | __/| _ | | _ || _ | _| _ | | ___| |__ |
* |___| |___._|__|__|_____||_____|__| |___._| |___| |__|_|__|_______|
*
* ============================================================================
* Copyright (c) 2007-2023 Artica Soluciones Tecnologicas, http://www.artica.es
* This code is NOT free software. This code is NOT licenced under GPL2 licence
* You cannnot redistribute it without written permission of copyright holder.
* ============================================================================
*/
use PandoraFMS\TacticalView\Element;
/**
* Alerts, this class contain all logic for this section.
*/
class Alerts extends Element
{
/**
* Constructor
*/
public function __construct()
{
parent::__construct();
$this->title = __('Alerts');
$this->ajaxMethods = ['getUsers'];
}
/**
* Returns the html of currently triggered.
*
* @return string
*/
public function getCurrentlyTriggered():string
{
// TODO connect to automonitorization.
return html_print_div(
[
'content' => '9.999.999',
'class' => 'text-l',
'style' => 'margin: 0px 10px 10px 10px;',
],
true
);
}
/**
* Returns the html of active correlation.
*
* @return string
*/
public function getActiveCorrelation():string
{
// TODO connect to automonitorization.
return html_print_div(
[
'content' => '9.999.999',
'class' => 'text-l',
'style' => 'margin: 0px 10px 10px 10px;',
],
true
);
}
/**
* Return a datatable with de users lists.
*
* @return string
*/
public function getDataTableUsers():string
{
$columns = [
'id_user',
'is_admin',
'last_connect',
];
$columnNames = [
__('User'),
__('Role'),
__('Last seen'),
];
return ui_print_datatable(
[
'id' => 'list_users',
'class' => 'info_table',
'style' => 'width: 90%',
'dom_elements' => 'tfp',
'filter_main_class' => 'box-flat white_table_graph fixed_filter_bar',
'columns' => $columns,
'column_names' => $columnNames,
'ajax_url' => $this->ajaxController,
'ajax_data' => [
'method' => 'getUsers',
'class' => static::class,
],
'order' => [
'field' => 'title',
'direction' => 'asc',
],
'default_pagination' => 8,
'search_button_class' => 'sub filter float-right',
'return' => true,
]
);
}
/**
* Return all users for ajax.
*
* @return void
*/
public function getUsers():void
{
global $config;
$start = get_parameter('start', 0);
$length = get_parameter('length', $config['block_size']);
$orderDatatable = get_datatable_order(true);
$pagination = '';
$order = '';
try {
ob_start();
if (isset($orderDatatable)) {
$order = sprintf(
' ORDER BY %s %s',
$orderDatatable['field'],
$orderDatatable['direction']
);
}
if (isset($length) && $length > 0
&& isset($start) && $start >= 0
) {
$pagination = sprintf(
' LIMIT %d OFFSET %d ',
$length,
$start
);
}
$sql = sprintf(
'SELECT id_user, is_admin ,last_connect
FROM tusuario u %s %s',
$order,
$pagination
);
$rows = db_process_sql($sql);
foreach ($rows as $key => $row) {
if ((bool) $row['is_admin'] === true) {
$rows[$key]['is_admin'] = '<span class="admin">'.__('Admin').'</span>';
} else {
$rows[$key]['is_admin'] = '<span class="user">'.__('User').'</span>';
}
if ($row['last_connect'] > 0) {
$rows[$key]['last_connect'] = ui_print_timestamp($row['last_connect'], true, ['prominent' => 'compact']);
} else {
$rows[$key]['last_connect'] = __('Unknown');
}
}
$sql_count = sprintf(
'SELECT count(*) as total FROM tusuario %s',
$order,
);
$total = db_process_sql($sql_count);
echo json_encode(
[
'data' => $rows,
'recordsTotal' => $total[0]['total'],
'recordsFiltered' => $total[0]['total'],
]
);
// Capture output.
$response = ob_get_clean();
} catch (Exception $e) {
echo json_encode(['error' => $e->getMessage()]);
exit;
}
json_decode($response);
if (json_last_error() === JSON_ERROR_NONE) {
echo $response;
} else {
echo json_encode(
[
'success' => false,
'error' => $response,
]
);
}
}
}

View File

@ -34,6 +34,7 @@ class Database extends Element
*/
public function __construct()
{
parent::__construct();
$this->title = __('Database');
}
@ -203,7 +204,7 @@ class Database extends Element
[
'content' => line_graph($data, $options),
'class' => 'w100p h100p',
'style' => 'max-height: 100px;',
'style' => 'max-height: 83px;',
],
true
);
@ -287,7 +288,7 @@ class Database extends Element
[
'content' => line_graph($data, $options),
'class' => 'w100p h100p',
'style' => 'max-height: 100px;',
'style' => 'max-height: 83px;',
],
true
);

View File

@ -0,0 +1,337 @@
<?php
/**
* Events element for tactical view.
*
* @category General
* @package Pandora FMS
* @subpackage TacticalView
* @version 1.0.0
* @license See below
*
* ______ ___ _______ _______ ________
* | __ \.-----.--.--.--| |.-----.----.-----. | ___| | | __|
* | __/| _ | | _ || _ | _| _ | | ___| |__ |
* |___| |___._|__|__|_____||_____|__| |___._| |___| |__|_|__|_______|
*
* ============================================================================
* Copyright (c) 2007-2023 Artica Soluciones Tecnologicas, http://www.artica.es
* This code is NOT free software. This code is NOT licenced under GPL2 licence
* You cannnot redistribute it without written permission of copyright holder.
* ============================================================================
*/
use PandoraFMS\TacticalView\Element;
/**
* Events, this class contain all logic for this section.
*/
class Events extends Element
{
/**
* Constructor
*/
public function __construct()
{
parent::__construct();
$this->title = __('Events');
}
/**
* Return the html graph of events in last 24h.
*
* @return string
*/
public function getEventsGraph():string
{
$interval24h = (time() - 86400);
$sql = 'SELECT
utimestamp,
DATE_FORMAT(FROM_UNIXTIME(utimestamp), "%Y-%m-%d %H:00:00") AS hour,
COUNT(*) AS number_of_events
FROM tevento
WHERE utimestamp >= '.$interval24h.'
GROUP BY hour
ORDER BY hour
LIMIT 24;';
$sqlTest = 'SELECT
utimestamp,
DATE_FORMAT(FROM_UNIXTIME(utimestamp), "%Y-%m-%d %H:00:00") AS hour,
COUNT(*) AS number_of_events
FROM tevento
WHERE utimestamp >= 1693296001
GROUP BY hour
ORDER BY hour
LIMIT 24;';
$rows = db_process_sql($sql);
$graph_values = [];
$colors = [];
$max_value = 0;
foreach ($rows as $key => $row) {
if ($max_value < $row['number_of_events']) {
$max_value = $row['number_of_events'];
}
$graph_values[] = [
'y' => $row['number_of_events'],
'x' => date('d-m-Y H:00:00', $row['utimestamp']),
];
}
$danger = $max_value;
$warning = ($max_value / 2);
$ok = ($max_value / 3);
foreach ($graph_values as $key => $value) {
if ($value['y'] >= $danger) {
$colors[] = '#EC7176';
}
if ($value['y'] >= $warning && $value['y'] < $danger) {
$colors[] = '#FCAB10';
}
if ($value['y'] < $ok) {
$colors[] = '#82B92E';
}
}
$options = [
'height' => 237,
'legend' => ['display' => false],
'scales' => [
'x' => [
'bounds' => 'data',
'grid' => ['display' => false],
'display' => false,
],
'y' => [
'grid' => ['display' => false],
],
],
'colors' => $colors,
'borderColors' => ['#ffffff'],
];
$bar = vbar_graph($graph_values, $options);
return html_print_div(
[
'content' => $bar,
'class' => 'margin-top-5 w100p relative',
'style' => 'max-height: 250px;',
],
true
);
}
/**
* Return the html graph of events in last 8h grouped by criticity.
*
* @return string
*/
public function getEventsCriticalityGraph():string
{
$interval8h = (time() - 86400);
$sql = 'SELECT criticity, count(*) AS total
FROM tevento
WHERE utimestamp >= '.$interval8h.'
group by criticity';
$sqlTest = 'SELECT criticity, count(*) AS total
FROM tevento
group by criticity';
$rows = db_process_sql($sql);
$labels = [];
$data = [];
$colors = [];
foreach ($rows as $key => $row) {
switch ($row['criticity']) {
case EVENT_CRIT_CRITICAL:
$label = __('CRITICAL');
$colors[] = COL_CRITICAL;
break;
case EVENT_CRIT_MAINTENANCE:
$label = __('MAINTENANCE');
$colors[] = COL_MAINTENANCE;
break;
case EVENT_CRIT_INFORMATIONAL:
$label = __('INFORMATIONAL');
$colors[] = COL_INFORMATIONAL;
break;
case EVENT_CRIT_MAJOR:
$label = __('MAJOR');
$colors[] = COL_MAJOR;
break;
case EVENT_CRIT_MINOR:
$label = __('MINOR');
$colors[] = COL_MINOR;
break;
case EVENT_CRIT_NORMAL:
$label = __('NORMAL');
$colors[] = COL_NORMAL;
break;
case EVENT_CRIT_WARNING:
$label = __('WARNING');
$colors[] = COL_WARNING;
break;
default:
$colors[] = COL_UNKNOWN;
$label = __('UNKNOWN');
break;
}
$labels[] = $this->controlSizeText($label);
$data[] = $row['total'];
}
$options = [
'labels' => $labels,
'legend' => ['display' => false],
'cutout' => 80,
'nodata_image' => ['width' => '100%'],
'colors' => $colors,
];
$pie = ring_graph($data, $options);
$output = html_print_div(
[
'content' => $pie,
'style' => 'margin: 0 auto; max-width: 80%; max-height: 220px;',
],
true
);
return $output;
}
/**
* Return the html graph of events in last 8h grouped by status validate.
*
* @return string
*/
public function getEventsStatusValidateGraph():string
{
$interval8h = (time() - 86400);
$sql = 'SELECT estado, count(*) AS total
FROM tevento
WHERE utimestamp >= '.$interval8h.'
group by estado';
$sqlTest = 'SELECT estado, count(*) AS total
FROM tevento
WHERE utimestamp <= 1688981702
group by estado';
$rows = db_process_sql($sql);
$labels = [];
$data = [];
foreach ($rows as $key => $row) {
switch ($row['estado']) {
case '2':
$label = _('In process');
break;
case '0':
$label = _('New events');
break;
case '3':
$label = _('Not validated');
break;
case '1':
$label = _('Validated events');
break;
default:
$label = __('Unknow');
break;
}
$labels[] = $label;
$data[] = $row['total'];
}
$options = [
'labels' => $labels,
'legend' => ['display' => false],
'cutout' => 80,
'nodata_image' => ['width' => '100%'],
];
$pie = ring_graph($data, $options);
$output = html_print_div(
[
'content' => $pie,
'style' => 'margin: 0 auto; max-width: 80%; max-height: 220px;',
],
true
);
return $output;
}
/**
* Return the datatable events in last 8 hours.
*
* @return string
*/
public function getDataTableEvents()
{
$column_names = [
__('S'),
__('Event'),
__('Date'),
];
$fields = [
'mini_severity',
'evento',
'timestamp',
];
return ui_print_datatable(
[
'id' => 'datatable_events',
'class' => 'info_table events',
'style' => 'width: 90%;',
'ajax_url' => 'operation/events/events',
'ajax_data' => [
'get_events' => 1,
'compact_date' => 1,
],
'order' => [
'field' => 'timestamp',
'direction' => 'desc',
],
'column_names' => $column_names,
'columns' => $fields,
'ajax_return_operation' => 'buffers',
'ajax_return_operation_function' => 'process_buffers',
'return' => true,
'csv' => 0,
'dom_elements' => 'tfp',
'default_pagination' => 8,
]
);
}
}

View File

@ -41,6 +41,7 @@ class Groups extends Element
*/
public function __construct()
{
parent::__construct();
$this->title = __('Groups');
$this->total = $this->calculateTotalGroups();
}
@ -67,7 +68,7 @@ class Groups extends Element
{
ui_require_css_file('heatmap');
$width = 350;
$height = 335;
$height = 275;
$sql = 'SELECT * FROM tagente a
LEFT JOIN tagent_secondary_group g ON g.id_agent = a.id_agente';
@ -205,7 +206,7 @@ class Groups extends Element
return html_print_div(
[
'content' => $heatmap,
'style' => 'margin: 0 auto; width: fit-content;',
'style' => 'margin: 0 auto; width: fit-content; min-height: 285px;',
],
true
);

View File

@ -34,6 +34,7 @@ class LogStorage extends Element
*/
public function __construct()
{
parent::__construct();
$this->title = __('Log storage');
}

View File

@ -34,6 +34,7 @@ class MonitoringElements extends Element
*/
public function __construct()
{
parent::__construct();
$this->title = __('Monitoring elements');
}
@ -48,7 +49,9 @@ class MonitoringElements extends Element
$sql = 'SELECT name, count(*) AS total
FROM ttag_module t
LEFT JOIN ttag ta ON ta.id_tag = t.id_tag
GROUP BY t.id_tag';
GROUP BY t.id_tag
ORDER BY total DESC
LIMIT 10;';
$rows = db_process_sql($sql);
$labels = [];
@ -63,6 +66,7 @@ class MonitoringElements extends Element
'legend' => [
'position' => 'bottom',
'align' => 'right',
'display' => false,
],
'cutout' => 80,
'nodata_image' => ['width' => '100%'],
@ -71,7 +75,7 @@ class MonitoringElements extends Element
$output = html_print_div(
[
'content' => $pie,
'style' => 'margin: 0 auto; max-width: 60%; max-height: 220px;',
'style' => 'margin: 0 auto; max-width: 80%; max-height: 220px;',
],
true
);
@ -91,7 +95,9 @@ class MonitoringElements extends Element
FROM tagente_modulo m
LEFT JOIN tmodule_group g ON g.id_mg = m.id_module_group
WHERE name <> ""
GROUP BY m.id_module_group';
GROUP BY m.id_module_group
ORDER BY total DESC
LIMIT 10';
$rows = db_process_sql($sql);
$labels = [];
@ -106,6 +112,7 @@ class MonitoringElements extends Element
'legend' => [
'position' => 'bottom',
'align' => 'right',
'display' => false,
],
'cutout' => 80,
'nodata_image' => ['width' => '100%'],
@ -114,7 +121,7 @@ class MonitoringElements extends Element
$output = html_print_div(
[
'content' => $pie,
'style' => 'margin: 0 auto; max-width: 60%; max-height: 220px;',
'style' => 'margin: 0 auto; max-width: 80%; max-height: 220px;',
],
true
);
@ -134,7 +141,9 @@ class MonitoringElements extends Element
FROM tagente a
LEFT JOIN tagent_secondary_group g ON g.id_agent = a.id_agente
LEFT JOIN tgrupo gr ON gr.id_grupo = a.id_grupo
GROUP BY a.id_grupo';
GROUP BY a.id_grupo
ORDER BY total DESC
LIMIT 10';
$rows = db_process_sql($sql);
$labels = [];
@ -149,6 +158,7 @@ class MonitoringElements extends Element
'legend' => [
'position' => 'bottom',
'align' => 'right',
'display' => false,
],
'cutout' => 80,
'nodata_image' => ['width' => '100%'],
@ -157,7 +167,7 @@ class MonitoringElements extends Element
$output = html_print_div(
[
'content' => $pie,
'style' => 'margin: 0 auto; max-width: 60%; max-height: 220px;',
'style' => 'margin: 0 auto; max-width: 80%; max-height: 220px;',
],
true
);
@ -178,7 +188,7 @@ class MonitoringElements extends Element
$output = html_print_div(
[
'content' => $pie,
'style' => 'margin: 0 auto; max-width: 60%; max-height: 220px;',
'style' => 'margin: 0 auto; max-width: 80%; max-height: 220px;',
],
true
);

View File

@ -34,6 +34,7 @@ class NewsBoard extends Element
*/
public function __construct()
{
parent::__construct();
ui_require_css_file('news');
include_once 'general/news_dialog.php';
$this->title = __('News Board');

View File

@ -34,6 +34,7 @@ class Overview extends Element
*/
public function __construct()
{
parent::__construct();
$this->title = __('General overview');
}

View File

@ -34,6 +34,7 @@ class SnmpTraps extends Element
*/
public function __construct()
{
parent::__construct();
$this->title = __('SNMP Traps');
}

View File

@ -20,9 +20,11 @@
.row {
display: flex;
width: 100%;
justify-content: space-between;
}
.col-6 {
width: 49%;
.col-6,
.col-xl-6 {
width: 50%;
display: flex;
flex-wrap: wrap;
flex-direction: column;
@ -30,8 +32,20 @@
.col-7 {
width: 58%;
}
.col-8 {
width: 66.6%;
}
.col-4 {
width: 33.3%;
}
.col-5 {
width: 41%;
width: 42%;
}
.col-3 {
width: 25%;
}
.col-9 {
width: 75%;
}
.container {
background-color: white;
@ -100,13 +114,14 @@
#news-board {
min-width: 530px;
width: 100%;
max-height: 805px;
max-height: 639px;
overflow-y: auto;
}
#database .subtitle,
#logStorage .subtitle,
#SNMPTraps .subtitle {
#SNMPTraps .subtitle,
#Alerts .subtitle {
padding: 10px 10px 5px 10px;
}
@ -117,3 +132,91 @@
text-align: left;
padding: 0px 10px 10px 10px;
}
.dataTables_paginate a {
background: none !important;
border: 0px;
font-size: 13px;
}
.pandora_pagination.next,
.pandora_pagination.previous {
border-right: 0px !important;
border-left: 0px !important;
}
a.pandora_pagination {
border-radius: 3px;
padding: 6px;
}
.info_table > tbody > tr:nth-child(even) {
background-color: #f9f9f9;
}
.pandora_pagination.current {
background-color: #1d7874 !important;
}
a.pandora_pagination:first-child {
border-left: 0px !important;
}
.info_table {
border: 1px solid #c0ccdc;
border-bottom: 1px solid #c0ccdc !important;
border-collapse: collapse;
}
.info_table td:nth-child(1) {
border-top: 0px;
}
.info_table td {
border-top: 1px solid #c0ccdc !important;
}
.info_table th {
border: 0px !important;
}
.admin {
color: #ec7176;
}
.user {
color: #8a96a6;
}
#Alerts {
padding-bottom: 20px;
}
.mini-criticity {
width: 6px !important;
min-width: auto !important;
max-width: auto !important;
height: 28px !important;
border-radius: 9px !important;
margin-left: 7px !important;
}
.datatables_thead_tr th:nth-child(1) {
width: 10.3333px !important;
padding: 20px 0px 20px 15px !important;
}
.info_table td {
font-size: 13px !important;
}
@media (max-width: 1636px) {
.col-xl-6 {
width: 100%;
}
.col-6 {
width: 49%;
}
.row {
flex-wrap: wrap;
}
}

View File

@ -340,6 +340,7 @@ if (is_ajax() === true) {
$get_events = (int) get_parameter('get_events', 0);
$table_id = get_parameter('table_id', '');
$groupRecursion = (bool) get_parameter('groupRecursion', false);
$compact_date = (int) get_parameter('compact_date', 0);
// Datatables offset, limit.
$start = (int) get_parameter('start', 0);
@ -471,7 +472,7 @@ if (is_ajax() === true) {
$data = array_reduce(
$events,
function ($carry, $item) use ($table_id, &$redirection_form_id, $filter) {
function ($carry, $item) use ($table_id, &$redirection_form_id, $filter, $compact_date) {
global $config;
$tmp = (object) $item;
@ -610,6 +611,12 @@ if (is_ajax() === true) {
);
$user_timezone = users_get_user_by_id($_SESSION['id_usuario'])['timezone'];
if ($compact_date === 1) {
$options = ['prominent' => 'compact'];
} else {
$options = [];
}
if (empty($user_timezone) === true) {
if (date_default_timezone_get() !== $config['timezone']) {
$timezone = timezone_open(date_default_timezone_get());
@ -624,16 +631,16 @@ if (is_ajax() === true) {
$total_sec = strtotime($tmp->timestamp);
$total_sec += $dif;
$last_contact = date($confb64ig['date_format'], $total_sec);
$last_contact_value = ui_print_timestamp($last_contact, true);
$last_contact_value = ui_print_timestamp($last_contact, true, $options);
} else {
$title = date($config['date_format'], strtotime($tmp->timestamp));
$value = ui_print_timestamp(strtotime($tmp->timestamp), true);
$value = ui_print_timestamp(strtotime($tmp->timestamp), true, $options);
$last_contact_value = '<span title="'.$title.'">'.$value.'</span>';
}
} else {
date_default_timezone_set($user_timezone);
$title = date($config['date_format'], strtotime($tmp->timestamp));
$value = ui_print_timestamp(strtotime($tmp->timestamp), true);
$value = ui_print_timestamp(strtotime($tmp->timestamp), true, $options);
$last_contact_value = '<span title="'.$title.'">'.$value.'</span>';
}