Implements live graphs page

main
Yasen Pramatarov 2025-01-14 15:37:20 +02:00
parent 5982d5eef9
commit 8b1fd2e2c1
3 changed files with 268 additions and 170 deletions

View File

@ -494,6 +494,77 @@ class Agent {
return null; return null;
} }
/**
* Gets historical data for a specific metric from agent checks
*
* @param int $platform_id The platform ID
* @param string $agent_type The type of agent (e.g., 'jvb', 'jicofo')
* @param string $metric_type The type of metric to retrieve
* @param string $from_time Start time in Y-m-d format
* @param string $until_time End time in Y-m-d format
* @return array Array with the dataset from agent checks
*/
public function getHistoricalData($platform_id, $agent_type, $metric_type, $from_time, $until_time) {
// Get data from agent checks
$sql = 'SELECT
DATE(jac.timestamp) as date,
jac.response_content,
COUNT(*) as checks_count
FROM
jilo_agent_checks jac
JOIN
jilo_agents ja ON jac.agent_id = ja.id
JOIN
jilo_agent_types jat ON ja.agent_type_id = jat.id
WHERE
ja.platform_id = :platform_id
AND jat.description = :agent_type
AND jac.status_code = 200
AND DATE(jac.timestamp) BETWEEN :from_time AND :until_time
GROUP BY
DATE(jac.timestamp)
ORDER BY
DATE(jac.timestamp)';
$query = $this->db->prepare($sql);
$query->execute([
':platform_id' => $platform_id,
':agent_type' => $agent_type,
':from_time' => $from_time,
':until_time' => $until_time
]);
$results = $query->fetchAll(PDO::FETCH_ASSOC);
$data = [];
foreach ($results as $row) {
$json_data = json_decode($row['response_content'], true);
if (json_last_error() === JSON_ERROR_NONE) {
$api_data = [];
if ($agent_type === 'jvb') {
$api_data = $json_data['jvb_api_data'] ?? [];
} elseif ($agent_type === 'jicofo') {
$api_data = $json_data['jicofo_api_data'] ?? [];
} elseif ($agent_type === 'jigasi') {
$api_data = $json_data['jigasi_api_data'] ?? [];
} elseif ($agent_type === 'prosody') {
$api_data = $json_data['prosody_api_data'] ?? [];
} elseif ($agent_type === 'nginx') {
$api_data = $json_data['nginx_api_data'] ?? [];
}
$value = $this->getNestedValue($api_data, $metric_type);
if ($value !== null) {
$data[] = [
'date' => $row['date'],
'value' => $value
];
}
}
}
return $data;
}
} }
?> ?>

View File

@ -5,40 +5,30 @@
<script> <script>
var ctx = document.getElementById('graph_<?= $data['graph_name'] ?>').getContext('2d'); var ctx = document.getElementById('graph_<?= $data['graph_name'] ?>').getContext('2d');
var chartData0 = <?php echo json_encode($data['data0']); ?>;
var chartData1 = <?php echo json_encode($data['data1']); ?>;
var timeRangeName = ''; var timeRangeName = '';
var labels = chartData0.map(function(item) { // Prepare datasets
return item.date; var datasets = [];
}); <?php foreach ($data['datasets'] as $dataset): ?>
var values0 = chartData0.map(function(item) { var chartData = <?php echo json_encode($dataset['data']); ?>;
return item.value; datasets.push({
}); label: '<?= $dataset['label'] ?>',
var values1 = chartData1.map(function(item) { data: chartData.map(function(item) {
return item.value; return {
}); x: item.date,
y: item.value
};
}),
borderColor: '<?= $dataset['color'] ?>',
borderWidth: 1,
fill: false
});
<?php endforeach; ?>
var graph_<?= $data['graph_name'] ?> = new Chart(ctx, { var graph_<?= $data['graph_name'] ?> = new Chart(ctx, {
type: 'line', type: 'line',
data: { data: {
labels: labels, datasets: datasets
datasets: [
{
label: '<?= $data['graph_data0_label'] ?? '' ?>',
data: values0,
borderColor: 'rgba(75, 192, 192, 1)',
borderWidth: 1,
fill: false
},
{
label: '<?= $data['graph_data1_label'] ?? '' ?>',
data: values1,
borderColor: 'rgba(255, 99, 132, 1)',
borderWidth: 1,
fill: false
}
]
}, },
options: { options: {
layout: { layout: {
@ -64,7 +54,6 @@ var graph_<?= $data['graph_name'] ?> = new Chart(ctx, {
mode: 'x' mode: 'x'
}, },
zoom: { zoom: {
// enabled: true,
mode: 'x', mode: 'x',
drag: { drag: {
enabled: true, // Enable drag to select range enabled: true, // Enable drag to select range
@ -99,29 +88,25 @@ var graph_<?= $data['graph_name'] ?> = new Chart(ctx, {
} }
}); });
// Store the graphs in an array // Store graph instance and title for later reference
graphs.push({ graphs.push({
graph: graph_<?= $data['graph_name'] ?>, graph: graph_<?= $data['graph_name'] ?>,
label: document.getElementById('current-period-<?= $data['graph_name'] ?>') label: '<?= $data['graph_title'] ?>'
}); });
// Update the time range label // Function to update the period label
function updatePeriodLabel(chart, labelElement) { function updatePeriodLabel(chart, label) {
var startDate = chart.scales.x.min; var startDate = new Date(chart.scales.x.min);
var endDate = chart.scales.x.max; var endDate = new Date(chart.scales.x.max);
if (timeRangeName == 'today') { var periodLabel = document.getElementById('current-period-<?= $data['graph_name'] ?>');
labelElement.innerHTML = 'Currently displaying: ' + timeRangeName + ' (' + new Date(startDate).toLocaleDateString() + ')';
if (timeRangeName) {
periodLabel.textContent = label + ' (' + timeRangeName + ')';
} else { } else {
labelElement.innerHTML = 'Currently displaying: ' + timeRangeName + ' (' + new Date(startDate).toLocaleDateString() + ' - ' + new Date(endDate).toLocaleDateString() + ')'; periodLabel.textContent = label + ' (from ' + startDate.toLocaleDateString() + ' to ' + endDate.toLocaleDateString() + ')';
} }
} }
// Attach the update function to the 'zoom' event // Initial label update
graph_<?= $data['graph_name'] ?>.options.plugins.zoom.onZoom = function({ chart }) { updatePeriodLabel(graph_<?= $data['graph_name'] ?>, '<?= $data['graph_title'] ?>');
updatePeriodLabel(chart, document.getElementById('current-period-<?= $data['graph_name'] ?>'));
};
// Update the label initially when the chart is rendered
updatePeriodLabel(graph_<?= $data['graph_name'] ?>, document.getElementById('current-period-<?= $data['graph_name'] ?>'));
</script> </script>

View File

@ -9,154 +9,196 @@ $agent = $_REQUEST['agent'] ?? '';
require '../app/classes/config.php'; require '../app/classes/config.php';
require '../app/classes/agent.php'; require '../app/classes/agent.php';
require '../app/classes/conference.php';
$configObject = new Config(); $configObject = new Config();
$agentObject = new Agent($dbWeb); $agentObject = new Agent($dbWeb);
switch ($item) { // connect to Jilo database
$response = connectDB($config, 'jilo', $platformDetails[0]['jilo_database'], $platform_id);
case 'graphs': // if DB connection has error, display it and stop here
// FIXME example data if ($response['db'] === null) {
$one = date('Y-m-d',strtotime("-5 days")); $error = $response['error'];
$two = date('Y-m-d',strtotime("-4 days")); include '../app/templates/block-message.php';
$three = date('Y-m-d',strtotime("-2 days"));
$four = date('Y-m-d',strtotime("-1 days"));
$graph[0]['data0'] = [ // otherwise if DB connection is OK, go on
['date' => $one, 'value' => 10], } else {
['date' => $two, 'value' => 20], $db = $response['db'];
['date' => $three, 'value' => 15],
['date' => $four, 'value' => 25],
];
$graph[0]['data1'] = [ $conferenceObject = new Conference($db);
['date' => $one, 'value' => 12],
['date' => $two, 'value' => 23],
['date' => $three, 'value' => 11],
['date' => $four, 'value' => 27],
];
$graph[0]['graph_name'] = 'conferences'; switch ($item) {
$graph[0]['graph_title'] = 'Conferences in "' . htmlspecialchars($platformDetails[0]['name']) . '" over time';
$graph[0]['graph_data0_label'] = 'Conferences from Jitsi logs (Jilo)';
$graph[0]['graph_data1_label'] = 'Conferences from Jitsi API (Jilo Agents)';
$graph[1]['data0'] = [ case 'graphs':
['date' => $one, 'value' => 20], // Connect to Jilo database for log data
['date' => $two, 'value' => 30], $jilo_response = connectDB($config, 'jilo', $platformDetails[0]['jilo_database'], $platform_id);
['date' => $three, 'value' => 15], if ($jilo_response['db'] === null) {
['date' => $four, 'value' => 55], $error = $jilo_response['error'];
]; include '../app/templates/block-message.php';
break;
}
$jilo_db = $jilo_response['db'];
$graph[1]['data1'] = [ // Get date range for the last 7 days
['date' => $one, 'value' => 22], $from_time = date('Y-m-d', strtotime('-7 days'));
['date' => $two, 'value' => 33], $until_time = date('Y-m-d');
['date' => $three, 'value' => 11],
['date' => $four, 'value' => 57],
];
$graph[1]['graph_name'] = 'participants'; // Define graphs to show
$graph[1]['graph_title'] = 'Participants in "' . htmlspecialchars($platformDetails[0]['name']) . '" over time'; $graphs = [
$graph[1]['graph_data0_label'] = 'Participants from Jitsi logs (Jilo)'; [
$graph[1]['graph_data1_label'] = 'Participants from Jitsi API (Jilo Agents)'; 'graph_name' => 'conferences',
'graph_title' => 'Conferences in "' . htmlspecialchars($platformDetails[0]['name']) . '" over time',
include '../app/templates/graphs-combined.php'; 'datasets' => []
break; ],
[
case 'latest': 'graph_name' => 'participants',
// Define metrics to display 'graph_title' => 'Participants in "' . htmlspecialchars($platformDetails[0]['name']) . '" over time',
$metrics = [ 'datasets' => []
'Basic stats' => [ ]
'conferences' => ['label' => 'Current conferences', 'link' => 'conferences'],
'participants' => ['label' => 'Current participants', 'link' => 'participants'],
'total_conferences_created' => ['label' => 'Total conferences created'],
'total_participants' => ['label' => 'Total participants']
],
'Bridge stats' => [
'bridge_selector.bridge_count' => ['label' => 'Bridge count'],
'bridge_selector.operational_bridge_count' => ['label' => 'Operational bridges'],
'bridge_selector.in_shutdown_bridge_count' => ['label' => 'Bridges in shutdown']
],
'Jibri stats' => [
'jibri_detector.count' => ['label' => 'Jibri count'],
'jibri_detector.available' => ['label' => 'Jibri idle'],
'jibri.live_streaming_active' => ['label' => 'Jibri active streaming'],
'jibri.recording_active' => ['label' => 'Jibri active recording'],
],
'System stats' => [
'threads' => ['label' => 'Threads'],
'stress_level' => ['label' => 'Stress level'],
'version' => ['label' => 'Version']
]
];
// Get latest data for all the agents
$agents = ['jvb', 'jicofo', 'jibri', 'prosody', 'nginx'];
$widget['records'] = [];
// Initialize records for each agent
foreach ($agents as $agent) {
$record = [
'table_headers' => strtoupper($agent),
'metrics' => [],
'timestamp' => null
]; ];
// Fetch all metrics for this agent // Get Jitsi API data
foreach ($metrics as $section => $section_metrics) { $conferences_api = $agentObject->getHistoricalData(
foreach ($section_metrics as $metric => $config) { $platform_id,
$data = $agentObject->getLatestData($platform_id, $agent, $metric); 'jicofo',
if ($data !== null) { 'conferences',
$record['metrics'][$section][$metric] = [ $from_time,
'value' => $data['value'], $until_time
'label' => $config['label'], );
'link' => isset($config['link']) ? $config['link'] : null $graphs[0]['datasets'][] = [
]; 'data' => $conferences_api,
// Use the most recent timestamp 'label' => 'Conferences from Jitsi API',
if ($record['timestamp'] === null || strtotime($data['timestamp']) > strtotime($record['timestamp'])) { 'color' => 'rgba(75, 192, 192, 1)'
$record['timestamp'] = $data['timestamp']; ];
// Get conference data from logs
$conferences_logs = $conferenceObject->conferenceNumber(
$from_time,
$until_time
);
$graphs[0]['datasets'][] = [
'data' => $conferences_logs,
'label' => 'Conferences from Logs',
'color' => 'rgba(255, 99, 132, 1)'
];
// Get participants data
$participants_api = $agentObject->getHistoricalData(
$platform_id,
'jicofo',
'participants',
$from_time,
$until_time
);
$graphs[1]['datasets'][] = [
'data' => $participants_api,
'label' => 'Participants from Jitsi API',
'color' => 'rgba(75, 192, 192, 1)'
];
// Prepare data for template
$graph = $graphs;
include '../app/templates/graphs-combined.php';
break;
case 'latest':
// Define metrics to display
$metrics = [
'Basic stats' => [
'conferences' => ['label' => 'Current conferences', 'link' => 'conferences'],
'participants' => ['label' => 'Current participants', 'link' => 'participants'],
'total_conferences_created' => ['label' => 'Total conferences created'],
'total_participants' => ['label' => 'Total participants']
],
'Bridge stats' => [
'bridge_selector.bridge_count' => ['label' => 'Bridge count'],
'bridge_selector.operational_bridge_count' => ['label' => 'Operational bridges'],
'bridge_selector.in_shutdown_bridge_count' => ['label' => 'Bridges in shutdown']
],
'Jibri stats' => [
'jibri_detector.count' => ['label' => 'Jibri count'],
'jibri_detector.available' => ['label' => 'Jibri idle'],
'jibri.live_streaming_active' => ['label' => 'Jibri active streaming'],
'jibri.recording_active' => ['label' => 'Jibri active recording'],
],
'System stats' => [
'threads' => ['label' => 'Threads'],
'stress_level' => ['label' => 'Stress level'],
'version' => ['label' => 'Version']
]
];
// Get latest data for all the agents
$agents = ['jvb', 'jicofo', 'jibri', 'prosody', 'nginx'];
$widget['records'] = [];
// Initialize records for each agent
foreach ($agents as $agent) {
$record = [
'table_headers' => strtoupper($agent),
'metrics' => [],
'timestamp' => null
];
// Fetch all metrics for this agent
foreach ($metrics as $section => $section_metrics) {
foreach ($section_metrics as $metric => $config) {
$data = $agentObject->getLatestData($platform_id, $agent, $metric);
if ($data !== null) {
$record['metrics'][$section][$metric] = [
'value' => $data['value'],
'label' => $config['label'],
'link' => isset($config['link']) ? $config['link'] : null
];
// Use the most recent timestamp
if ($record['timestamp'] === null || strtotime($data['timestamp']) > strtotime($record['timestamp'])) {
$record['timestamp'] = $data['timestamp'];
}
} }
} }
} }
if (!empty($record['metrics'])) {
$widget['records'][] = $record;
}
} }
if (!empty($record['metrics'])) { // prepare the widget
$widget['records'][] = $record; $widget['full'] = false;
$widget['name'] = 'LatestData';
$widget['title'] = 'Latest data from Jilo Agents';
$widget['collapsible'] = false;
$widget['collapsed'] = false;
$widget['filter'] = false;
$widget['metrics'] = $metrics; // Pass metrics configuration to template
if (!empty($widget['records'])) {
$widget['full'] = true;
} }
} $widget['pagination'] = false;
// prepare the widget include '../app/templates/latest-data.php';
$widget['full'] = false; break;
$widget['name'] = 'LatestData';
$widget['title'] = 'Latest data from Jilo Agents';
$widget['collapsible'] = false;
$widget['collapsed'] = false;
$widget['filter'] = false;
$widget['metrics'] = $metrics; // Pass metrics configuration to template
if (!empty($widget['records'])) {
$widget['full'] = true;
}
$widget['pagination'] = false;
include '../app/templates/latest-data.php'; case 'configjs':
break; $mode = $_REQUEST['mode'] ?? '';
$raw = ($mode === 'raw');
$platformConfigjs = $configObject->getPlatformConfigjs($platformDetails[0]['jitsi_url'], $raw);
include '../app/templates/data-configjs.php';
break;
case 'configjs': case 'interfaceconfigjs':
$mode = $_REQUEST['mode'] ?? ''; $mode = $_REQUEST['mode'] ?? '';
$raw = ($mode === 'raw'); $raw = ($mode === 'raw');
$platformConfigjs = $configObject->getPlatformConfigjs($platformDetails[0]['jitsi_url'], $raw); $platformInterfaceConfigjs = $configObject->getPlatformInterfaceConfigjs($platformDetails[0]['jitsi_url'], $raw);
include '../app/templates/data-configjs.php'; include '../app/templates/data-interfaceconfigjs.php';
break; break;
case 'interfaceconfigjs': default:
$mode = $_REQUEST['mode'] ?? ''; }
$raw = ($mode === 'raw');
$platformInterfaceConfigjs = $configObject->getPlatformInterfaceConfigjs($platformDetails[0]['jitsi_url'], $raw);
include '../app/templates/data-interfaceconfigjs.php';
break;
default:
} }
?> ?>