jilo-web/app/classes/agent.php

571 lines
18 KiB
PHP
Raw Permalink Normal View History

2024-09-04 22:06:38 +00:00
<?php
2024-11-29 17:02:40 +00:00
/**
* class Agent
*
* Provides methods to interact with Jilo agents, including retrieving details, managing agents, generating JWT tokens,
* and fetching data from agent APIs.
*/
2024-09-04 22:06:38 +00:00
class Agent {
2024-11-29 17:02:40 +00:00
/**
* @var PDO|null $db The database connection instance.
*/
2024-09-04 22:06:38 +00:00
private $db;
2024-11-29 17:02:40 +00:00
/**
* Agent constructor.
* Initializes the database connection.
*
* @param object $database The database object to initialize the connection.
*/
2024-09-04 22:06:38 +00:00
public function __construct($database) {
$this->db = $database->getConnection();
}
2024-11-29 17:02:40 +00:00
/**
* Retrieves details of a specified agent ID (or all agents) in a specified platform.
*
* @param int $platform_id The platform ID to filter agents by.
* @param int $agent_id The agent ID to filter by. If empty, all agents are returned.
*
* @return array The list of agent details.
*/
2024-09-04 22:06:38 +00:00
public function getAgentDetails($platform_id, $agent_id = '') {
2024-09-30 08:55:23 +00:00
$sql = 'SELECT
ja.id,
ja.platform_id,
ja.agent_type_id,
ja.url,
ja.secret_key,
2024-10-24 09:11:35 +00:00
ja.check_period,
2024-09-30 08:55:23 +00:00
jat.description AS agent_description,
jat.endpoint AS agent_endpoint
FROM
jilo_agents ja
JOIN
jilo_agent_types jat ON ja.agent_type_id = jat.id
2024-09-04 22:06:38 +00:00
WHERE
platform_id = :platform_id';
2024-09-30 08:55:23 +00:00
2024-09-04 22:06:38 +00:00
if ($agent_id !== '') {
2024-09-30 08:55:23 +00:00
$sql .= ' AND ja.id = :agent_id';
}
$query = $this->db->prepare($sql);
$query->bindParam(':platform_id', $platform_id);
if ($agent_id !== '') {
$query->bindParam(':agent_id', $agent_id);
2024-09-04 22:06:38 +00:00
}
$query->execute();
return $query->fetchAll(PDO::FETCH_ASSOC);
}
2024-11-29 17:02:40 +00:00
/**
* Retrieves details of a specified agent by its agent ID.
*
* @param int $agent_id The agent ID to filter by.
*
* @return array The agent details.
*/
public function getAgentIDDetails($agent_id) {
$sql = 'SELECT
ja.id,
ja.platform_id,
ja.agent_type_id,
ja.url,
ja.secret_key,
2024-10-24 09:11:35 +00:00
ja.check_period,
jat.description AS agent_description,
jat.endpoint AS agent_endpoint
FROM
jilo_agents ja
JOIN
jilo_agent_types jat ON ja.agent_type_id = jat.id
WHERE
ja.id = :agent_id';
$query = $this->db->prepare($sql);
$query->bindParam(':agent_id', $agent_id);
$query->execute();
return $query->fetchAll(PDO::FETCH_ASSOC);
}
2024-11-29 17:02:40 +00:00
/**
* Retrieves all agent types.
*
* @return array List of all agent types.
*/
public function getAgentTypes() {
$sql = 'SELECT *
FROM jilo_agent_types
ORDER BY id';
$query = $this->db->prepare($sql);
$query->execute();
return $query->fetchAll(PDO::FETCH_ASSOC);
}
2024-11-29 17:02:40 +00:00
/**
* Retrieves agent types already configured for a specific platform.
*
* @param int $platform_id The platform ID to filter agents by.
*
* @return array List of agent types configured for the platform.
*/
public function getPlatformAgentTypes($platform_id) {
$sql = 'SELECT
id,
agent_type_id
FROM
jilo_agents
WHERE
platform_id = :platform_id';
$query = $this->db->prepare($sql);
$query->bindParam(':platform_id', $platform_id);
$query->execute();
return $query->fetchAll(PDO::FETCH_ASSOC);
}
2024-11-29 17:02:40 +00:00
/**
* Adds a new agent to the platform.
*
* @param int $platform_id The platform ID where the agent is to be added.
* @param array $newAgent The new agent details to add.
*
* @return bool|string Returns true on success or an error message on failure.
*/
2024-09-04 22:06:38 +00:00
public function addAgent($platform_id, $newAgent) {
try {
$sql = 'INSERT INTO jilo_agents
(platform_id, agent_type_id, url, secret_key, check_period)
2024-09-04 22:06:38 +00:00
VALUES
(:platform_id, :agent_type_id, :url, :secret_key, :check_period)';
2024-09-04 22:06:38 +00:00
$query = $this->db->prepare($sql);
$query->execute([
':platform_id' => $platform_id,
':agent_type_id' => $newAgent['type_id'],
2024-09-04 22:06:38 +00:00
':url' => $newAgent['url'],
':secret_key' => $newAgent['secret_key'],
':check_period' => $newAgent['check_period'],
2024-09-04 22:06:38 +00:00
]);
return true;
} catch (Exception $e) {
return $e->getMessage();
}
}
2024-11-29 17:02:40 +00:00
/**
* Edits an existing agent's details.
*
* @param int $platform_id The platform ID where the agent exists.
* @param array $updatedAgent The updated agent details.
*
* @return bool|string Returns true on success or an error message on failure.
*/
2024-09-04 22:06:38 +00:00
public function editAgent($platform_id, $updatedAgent) {
try {
$sql = 'UPDATE jilo_agents SET
agent_type_id = :agent_type_id,
2024-09-04 22:06:38 +00:00
url = :url,
2024-10-24 09:11:35 +00:00
secret_key = :secret_key,
check_period = :check_period
2024-09-04 22:06:38 +00:00
WHERE
id = :agent_id
AND
platform_id = :platform_id';
$query = $this->db->prepare($sql);
$query->execute([
':agent_type_id' => $updatedAgent['agent_type_id'],
2024-09-04 22:06:38 +00:00
':url' => $updatedAgent['url'],
':secret_key' => $updatedAgent['secret_key'],
2024-10-24 09:11:35 +00:00
':check_period' => $updatedAgent['check_period'],
2024-09-04 22:06:38 +00:00
':agent_id' => $updatedAgent['id'],
':platform_id' => $platform_id,
]);
return true;
} catch (Exception $e) {
return $e->getMessage();
}
}
2024-11-29 17:02:40 +00:00
/**
* Deletes an agent from the platform.
*
* @param int $agent_id The agent ID to delete.
*
* @return bool|string Returns true on success or an error message on failure.
*/
2024-09-04 22:06:38 +00:00
public function deleteAgent($agent_id) {
try {
$sql = 'DELETE FROM jilo_agents
WHERE
id = :agent_id';
$query = $this->db->prepare($sql);
$query->bindParam(':agent_id', $agent_id);
$query->execute();
return true;
} catch (Exception $e) {
return $e->getMessage();
}
}
2024-10-01 07:18:53 +00:00
2024-11-29 17:02:40 +00:00
/**
* Checks if the agent cache is still valid.
*
* @param int $agent_id The agent ID to check.
*
* @return bool Returns true if cache is valid, false otherwise.
*/
2024-09-26 06:56:24 +00:00
public function checkAgentCache($agent_id) {
2024-10-03 07:59:32 +00:00
$agent_cache_name = 'agent' . $agent_id . '_cache';
$agent_cache_time = 'agent' . $agent_id . '_time';
2024-09-26 06:56:24 +00:00
return isset($_SESSION[$agent_cache_name]) && isset($_SESSION[$agent_cache_time]) && (time() - $_SESSION[$agent_cache_time] < 600);
}
2024-09-04 22:06:38 +00:00
2024-10-01 07:18:53 +00:00
2024-11-29 17:02:40 +00:00
/**
* Base64 URL encodes the input data. Used for encoding JWT tokens
*
* @param string $data The data to encode.
*
* @return string The base64 URL encoded string.
*/
2024-10-01 07:18:53 +00:00
private function base64UrlEncode($data) {
return rtrim(strtr(base64_encode($data), '+/', '-_'), '=');
}
2024-11-29 17:02:40 +00:00
/**
* Generates a JWT token for a Jilo agent.
*
* @param array $payload The payload data to include in the token.
* @param string $secret_key The secret key used to sign the token.
*
* @return string The generated JWT token.
*/
2024-10-01 07:18:53 +00:00
public function generateAgentToken($payload, $secret_key) {
// header
$header = json_encode([
'typ' => 'JWT',
'alg' => 'HS256'
]);
$base64Url_header = $this->base64UrlEncode($header);
// payload
$payload = json_encode($payload);
$base64Url_payload = $this->base64UrlEncode($payload);
// signature
$signature = hash_hmac('sha256', $base64Url_header . "." . $base64Url_payload, $secret_key, true);
$base64Url_signature = $this->base64UrlEncode($signature);
// build the JWT
$jwt = $base64Url_header . "." . $base64Url_payload . "." . $base64Url_signature;
return $jwt;
}
2024-11-29 17:02:40 +00:00
/**
* Fetches data from a Jilo agent's API, optionally forcing a refresh of the cache.
*
* @param int $agent_id The agent ID to fetch data for.
* @param bool $force Whether to force-refresh the cache (default: false).
*
* @return string The API response, or an error message in JSON format.
*/
2024-09-30 07:07:50 +00:00
public function fetchAgent($agent_id, $force = false) {
2024-09-28 07:04:20 +00:00
// we need agent details for URL and JWT token
$agentDetails = $this->getAgentIDDetails($agent_id);
// Safe exit in case the agent is not found
if (empty($agentDetails)) {
return json_encode(['error' => 'Agent not found']);
}
$agent = $agentDetails[0];
2024-10-03 07:59:32 +00:00
$agent_cache_name = 'agent' . $agent_id . '_cache';
$agent_cache_time = 'agent' . $agent_id . '_time';
2024-09-28 07:04:20 +00:00
// check if the cache is still valid, unless force-refresh is requested
if (!$force && $this->checkAgentCache($agent_id)) {
2024-09-28 07:04:20 +00:00
return $_SESSION[$agent_cache_name];
}
// generate the JWT token
$payload = [
'agent_id' => $agent_id,
'timestamp' => time()
];
$jwt = $this->generateAgentToken($payload, $agent['secret_key']);
2024-09-28 07:04:20 +00:00
// Make the API request
$ch = curl_init();
curl_setopt($ch, CURLOPT_URL, $agent['url'] . $agent['agent_endpoint']);
2024-09-28 07:04:20 +00:00
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
curl_setopt($ch, CURLOPT_TIMEOUT, 10); // timeout 10 seconds
curl_setopt($ch, CURLOPT_HTTPHEADER, [
'Authorization: Bearer ' . $jwt,
'Content-Type: application/json'
]);
2024-09-28 07:04:20 +00:00
$response = curl_exec($ch);
2024-09-30 08:55:23 +00:00
$curl_error = curl_error($ch);
$curl_errno = curl_errno($ch);
$http_code = curl_getinfo($ch, CURLINFO_HTTP_CODE);
2024-09-28 07:04:20 +00:00
curl_close($ch);
// curl error
if ($curl_errno) {
2024-09-30 08:55:23 +00:00
return json_encode(['error' => 'curl error: ' . $curl_error]);
}
// response is not 200 OK
if ($http_code !== 200) {
return json_encode(['error' => 'HTTP error: ' . $http_code]);
}
2024-09-30 08:55:23 +00:00
// other custom error(s)
if (strpos($response, 'Auth header not received') !== false) {
return json_encode(['error' => 'Auth header not received']);
2024-09-28 07:04:20 +00:00
}
2024-09-30 08:55:23 +00:00
// Cache the result and the timestamp if the response is successful
2024-10-23 10:22:55 +00:00
// We decode it so that it's pure JSON and not escaped
$_SESSION[$agent_cache_name] = json_decode($response, true);
2024-09-30 08:55:23 +00:00
$_SESSION[$agent_cache_time] = time();
2024-09-28 07:04:20 +00:00
return $response;
}
2024-11-29 17:02:40 +00:00
/**
* Clears the cached data for a specific agent.
*
* @param int $agent_id The agent ID for which the cache should be cleared.
*/
2024-10-03 07:59:32 +00:00
public function clearAgentCache($agent_id) {
$_SESSION["agent{$agent_id}_cache"] = '';
$_SESSION["agent{$agent_id}_cache_time"] = '';
}
2024-10-11 07:58:08 +00:00
2025-01-14 12:09:28 +00:00
/**
* Gets a value from a nested array using dot notation
* e.g. "bridge_selector.bridge_count" will get $array['bridge_selector']['bridge_count']
*
* @param array $array The array to search in
* @param string $path The path in dot notation
* @return mixed|null The value if found, null otherwise
*/
private function getNestedValue($array, $path) {
$keys = explode('.', $path);
$value = $array;
foreach ($keys as $key) {
if (!isset($value[$key])) {
return null;
}
$value = $value[$key];
}
return $value;
}
2024-11-29 17:02:40 +00:00
/**
* Retrieves the latest stored data for a specific platform, agent type, and metric type.
*
* @param int $platform_id The platform ID.
* @param string $agent_type The agent type.
* @param string $metric_type The metric type to filter by.
*
* @return mixed The latest stored data.
*/
2024-10-11 07:58:08 +00:00
public function getLatestData($platform_id, $agent_type, $metric_type) {
2025-01-14 12:09:28 +00:00
$sql = 'SELECT
2025-01-13 15:54:42 +00:00
jac.timestamp,
jac.response_content,
jac.agent_id,
jat.description
2025-01-14 12:09:28 +00:00
FROM
2025-01-13 15:54:42 +00:00
jilo_agent_checks jac
2025-01-14 12:09:28 +00:00
JOIN
2025-01-13 15:54:42 +00:00
jilo_agents ja ON jac.agent_id = ja.id
2025-01-14 12:09:28 +00:00
JOIN
2025-01-13 15:54:42 +00:00
jilo_agent_types jat ON ja.agent_type_id = jat.id
2025-01-14 12:09:28 +00:00
WHERE
2025-01-13 15:54:42 +00:00
ja.platform_id = :platform_id
AND jat.description = :agent_type
AND jac.status_code = 200
2025-01-14 12:09:28 +00:00
ORDER BY
2025-01-13 15:54:42 +00:00
jac.timestamp DESC
LIMIT 1';
$query = $this->db->prepare($sql);
$query->execute([
':platform_id' => $platform_id,
':agent_type' => $agent_type
]);
$result = $query->fetch(PDO::FETCH_ASSOC);
2025-01-14 12:09:28 +00:00
2025-01-13 15:54:42 +00:00
if ($result) {
// Parse the JSON response content
$data = json_decode($result['response_content'], true);
if (json_last_error() !== JSON_ERROR_NONE) {
return null;
}
2025-01-14 12:09:28 +00:00
2025-01-13 15:54:42 +00:00
// Extract the specific metric value from the response based on agent type
if ($agent_type === 'jvb') {
2025-01-14 12:09:28 +00:00
$value = $this->getNestedValue($data['jvb_api_data'], $metric_type);
if ($value !== null) {
2025-01-13 15:54:42 +00:00
return [
2025-01-14 12:09:28 +00:00
'value' => $value,
2025-01-13 15:54:42 +00:00
'timestamp' => $result['timestamp']
];
}
2025-01-14 12:09:28 +00:00
2025-01-13 15:54:42 +00:00
} elseif ($agent_type === 'jicofo') {
2025-01-14 12:09:28 +00:00
$value = $this->getNestedValue($data['jicofo_api_data'], $metric_type);
if ($value !== null) {
2025-01-13 15:54:42 +00:00
return [
2025-01-14 12:09:28 +00:00
'value' => $value,
'timestamp' => $result['timestamp']
];
}
} elseif ($agent_type === 'jigasi') {
$value = $this->getNestedValue($data['jigasi_api_data'], $metric_type);
if ($value !== null) {
return [
'value' => $value,
'timestamp' => $result['timestamp']
];
}
} elseif ($agent_type === 'prosody') {
$value = $this->getNestedValue($data['prosody_api_data'], $metric_type);
if ($value !== null) {
return [
'value' => $value,
'timestamp' => $result['timestamp']
];
}
} elseif ($agent_type === 'nginx') {
$value = $this->getNestedValue($data['nginx_api_data'], $metric_type);
if ($value !== null) {
return [
'value' => $value,
2025-01-13 15:54:42 +00:00
'timestamp' => $result['timestamp']
];
}
}
2025-01-14 12:09:28 +00:00
2025-01-13 15:54:42 +00:00
}
2025-01-14 12:09:28 +00:00
2025-01-13 15:54:42 +00:00
return null;
2024-10-11 07:58:08 +00:00
}
2025-01-14 13:37:20 +00:00
/**
* 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;
}
2024-09-04 22:06:38 +00:00
}
?>