314 lines
11 KiB
PHP
314 lines
11 KiB
PHP
<?php
|
|
|
|
use Utopia\App;
|
|
use Utopia\Exception;
|
|
use Utopia\Storage\Device\Local;
|
|
use Utopia\Storage\Storage;
|
|
use Appwrite\ClamAV\Network;
|
|
use Appwrite\Event\Event;
|
|
|
|
App::get('/v1/health')
|
|
->desc('Get HTTP')
|
|
->groups(['api', 'health'])
|
|
->label('scope', 'health.read')
|
|
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
|
|
->label('sdk.namespace', 'health')
|
|
->label('sdk.method', 'get')
|
|
->label('sdk.description', '/docs/references/health/get.md')
|
|
->inject('response')
|
|
->action(function ($response) {
|
|
/** @var Appwrite\Utopia\Response $response */
|
|
|
|
$response->json(['status' => 'OK']);
|
|
});
|
|
|
|
App::get('/v1/health/version')
|
|
->desc('Get Version')
|
|
->groups(['api', 'health'])
|
|
->label('scope', 'public')
|
|
->inject('response')
|
|
->action(function ($response) {
|
|
/** @var Appwrite\Utopia\Response $response */
|
|
|
|
$response->json(['version' => APP_VERSION_STABLE]);
|
|
});
|
|
|
|
App::get('/v1/health/db')
|
|
->desc('Get DB')
|
|
->groups(['api', 'health'])
|
|
->label('scope', 'health.read')
|
|
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
|
|
->label('sdk.namespace', 'health')
|
|
->label('sdk.method', 'getDB')
|
|
->label('sdk.description', '/docs/references/health/get-db.md')
|
|
->inject('response')
|
|
->inject('app')
|
|
->action(function ($response, $app) {
|
|
/** @var Appwrite\Utopia\Response $response */
|
|
/** @var Utopia\App $app */
|
|
$app->getResource('db');
|
|
|
|
$response->json(['status' => 'OK']);
|
|
});
|
|
|
|
App::get('/v1/health/cache')
|
|
->desc('Get Cache')
|
|
->groups(['api', 'health'])
|
|
->label('scope', 'health.read')
|
|
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
|
|
->label('sdk.namespace', 'health')
|
|
->label('sdk.method', 'getCache')
|
|
->label('sdk.description', '/docs/references/health/get-cache.md')
|
|
->inject('response')
|
|
->inject('app')
|
|
->action(function ($response, $app) {
|
|
/** @var Appwrite\Utopia\Response $response */
|
|
/** @var Utopia\App $register */
|
|
$app->getResource('cache');
|
|
|
|
$response->json(['status' => 'OK']);
|
|
});
|
|
|
|
App::get('/v1/health/time')
|
|
->desc('Get Time')
|
|
->groups(['api', 'health'])
|
|
->label('scope', 'health.read')
|
|
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
|
|
->label('sdk.namespace', 'health')
|
|
->label('sdk.method', 'getTime')
|
|
->label('sdk.description', '/docs/references/health/get-time.md')
|
|
->inject('response')
|
|
->action(function ($response) {
|
|
/** @var Appwrite\Utopia\Response $response */
|
|
|
|
/*
|
|
* Code from: @see https://www.beliefmedia.com.au/query-ntp-time-server
|
|
*/
|
|
$host = 'time.google.com'; // https://developers.google.com/time/
|
|
$gap = 60; // Allow [X] seconds gap
|
|
|
|
/* Create a socket and connect to NTP server */
|
|
$sock = \socket_create(AF_INET, SOCK_DGRAM, SOL_UDP);
|
|
|
|
\socket_connect($sock, $host, 123);
|
|
|
|
/* Send request */
|
|
$msg = "\010".\str_repeat("\0", 47);
|
|
|
|
\socket_send($sock, $msg, \strlen($msg), 0);
|
|
|
|
/* Receive response and close socket */
|
|
\socket_recv($sock, $recv, 48, MSG_WAITALL);
|
|
\socket_close($sock);
|
|
|
|
/* Interpret response */
|
|
$data = \unpack('N12', $recv);
|
|
$timestamp = \sprintf('%u', $data[9]);
|
|
|
|
/* NTP is number of seconds since 0000 UT on 1 January 1900
|
|
Unix time is seconds since 0000 UT on 1 January 1970 */
|
|
$timestamp -= 2208988800;
|
|
|
|
$diff = ($timestamp - \time());
|
|
|
|
if ($diff > $gap || $diff < ($gap * -1)) {
|
|
throw new Exception('Server time gaps detected');
|
|
}
|
|
|
|
$response->json(['remote' => $timestamp, 'local' => \time(), 'diff' => $diff]);
|
|
});
|
|
|
|
App::get('/v1/health/queue/webhooks')
|
|
->desc('Get Webhooks Queue')
|
|
->groups(['api', 'health'])
|
|
->label('scope', 'health.read')
|
|
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
|
|
->label('sdk.namespace', 'health')
|
|
->label('sdk.method', 'getQueueWebhooks')
|
|
->label('sdk.description', '/docs/references/health/get-queue-webhooks.md')
|
|
->inject('response')
|
|
->action(function ($response) {
|
|
/** @var Appwrite\Utopia\Response $response */
|
|
|
|
$response->json(['size' => Resque::size(Event::WEBHOOK_QUEUE_NAME)]);
|
|
}, ['response']);
|
|
|
|
App::get('/v1/health/queue/tasks')
|
|
->desc('Get Tasks Queue')
|
|
->groups(['api', 'health'])
|
|
->label('scope', 'health.read')
|
|
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
|
|
->label('sdk.namespace', 'health')
|
|
->label('sdk.method', 'getQueueTasks')
|
|
->label('sdk.description', '/docs/references/health/get-queue-tasks.md')
|
|
->inject('response')
|
|
->action(function ($response) {
|
|
/** @var Appwrite\Utopia\Response $response */
|
|
|
|
$response->json(['size' => Resque::size(Event::TASK_QUEUE_NAME)]);
|
|
}, ['response']);
|
|
|
|
App::get('/v1/health/queue/logs')
|
|
->desc('Get Logs Queue')
|
|
->groups(['api', 'health'])
|
|
->label('scope', 'health.read')
|
|
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
|
|
->label('sdk.namespace', 'health')
|
|
->label('sdk.method', 'getQueueLogs')
|
|
->label('sdk.description', '/docs/references/health/get-queue-logs.md')
|
|
->inject('response')
|
|
->action(function ($response) {
|
|
/** @var Appwrite\Utopia\Response $response */
|
|
|
|
$response->json(['size' => Resque::size(Event::AUDITS_QUEUE_NAME)]);
|
|
}, ['response']);
|
|
|
|
App::get('/v1/health/queue/usage')
|
|
->desc('Get Usage Queue')
|
|
->groups(['api', 'health'])
|
|
->label('scope', 'health.read')
|
|
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
|
|
->label('sdk.namespace', 'health')
|
|
->label('sdk.method', 'getQueueUsage')
|
|
->label('sdk.description', '/docs/references/health/get-queue-usage.md')
|
|
->inject('response')
|
|
->action(function ($response) {
|
|
/** @var Appwrite\Utopia\Response $response */
|
|
|
|
$response->json(['size' => Resque::size(Event::USAGE_QUEUE_NAME)]);
|
|
}, ['response']);
|
|
|
|
App::get('/v1/health/queue/certificates')
|
|
->desc('Get Certificate Queue')
|
|
->groups(['api', 'health'])
|
|
->label('scope', 'health.read')
|
|
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
|
|
->label('sdk.namespace', 'health')
|
|
->label('sdk.method', 'getQueueCertificates')
|
|
->label('sdk.description', '/docs/references/health/get-queue-certificates.md')
|
|
->inject('response')
|
|
->action(function ($response) {
|
|
/** @var Appwrite\Utopia\Response $response */
|
|
|
|
$response->json(['size' => Resque::size(Event::CERTIFICATES_QUEUE_NAME)]);
|
|
}, ['response']);
|
|
|
|
App::get('/v1/health/queue/functions')
|
|
->desc('Get Functions Queue')
|
|
->groups(['api', 'health'])
|
|
->label('scope', 'health.read')
|
|
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
|
|
->label('sdk.namespace', 'health')
|
|
->label('sdk.method', 'getQueueFunctions')
|
|
->label('sdk.description', '/docs/references/health/get-queue-functions.md')
|
|
->inject('response')
|
|
->action(function ($response) {
|
|
/** @var Appwrite\Utopia\Response $response */
|
|
|
|
$response->json(['size' => Resque::size(Event::FUNCTIONS_QUEUE_NAME)]);
|
|
}, ['response']);
|
|
|
|
App::get('/v1/health/storage/local')
|
|
->desc('Get Local Storage')
|
|
->groups(['api', 'health'])
|
|
->label('scope', 'health.read')
|
|
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
|
|
->label('sdk.namespace', 'health')
|
|
->label('sdk.method', 'getStorageLocal')
|
|
->label('sdk.description', '/docs/references/health/get-storage-local.md')
|
|
->inject('response')
|
|
->action(function ($response) {
|
|
/** @var Appwrite\Utopia\Response $response */
|
|
|
|
foreach ([
|
|
'Uploads' => APP_STORAGE_UPLOADS,
|
|
'Cache' => APP_STORAGE_CACHE,
|
|
'Config' => APP_STORAGE_CONFIG,
|
|
'Certs' => APP_STORAGE_CERTIFICATES
|
|
] as $key => $volume) {
|
|
$device = new Local($volume);
|
|
|
|
if (!\is_readable($device->getRoot())) {
|
|
throw new Exception('Device '.$key.' dir is not readable');
|
|
}
|
|
|
|
if (!\is_writable($device->getRoot())) {
|
|
throw new Exception('Device '.$key.' dir is not writable');
|
|
}
|
|
}
|
|
|
|
$response->json(['status' => 'OK']);
|
|
});
|
|
|
|
App::get('/v1/health/anti-virus')
|
|
->desc('Get Anti virus')
|
|
->groups(['api', 'health'])
|
|
->label('scope', 'health.read')
|
|
->label('sdk.auth', [APP_AUTH_TYPE_KEY])
|
|
->label('sdk.namespace', 'health')
|
|
->label('sdk.method', 'getAntiVirus')
|
|
->label('sdk.description', '/docs/references/health/get-storage-anti-virus.md')
|
|
->inject('response')
|
|
->action(function ($response) {
|
|
/** @var Appwrite\Utopia\Response $response */
|
|
|
|
if (App::getEnv('_APP_STORAGE_ANTIVIRUS') === 'disabled') { // Check if scans are enabled
|
|
return $response->json([
|
|
'status' => 'disabled',
|
|
'version' => '',
|
|
]);
|
|
}
|
|
|
|
$antiVirus = new Network(App::getEnv('_APP_STORAGE_ANTIVIRUS_HOST', 'clamav'),
|
|
(int) App::getEnv('_APP_STORAGE_ANTIVIRUS_PORT', 3310));
|
|
try {
|
|
$response->json([
|
|
'status' => (@$antiVirus->ping()) ? 'online' : 'offline',
|
|
'version' => @$antiVirus->version(),
|
|
]);
|
|
} catch (Throwable $e) {
|
|
$response->json([
|
|
'status' => 'offline',
|
|
'version' => '',
|
|
]);
|
|
}
|
|
});
|
|
|
|
App::get('/v1/health/stats') // Currently only used internally
|
|
->desc('Get System Stats')
|
|
->groups(['api', 'health'])
|
|
->label('scope', 'root')
|
|
// ->label('sdk.auth', [APP_AUTH_TYPE_KEY])
|
|
// ->label('sdk.namespace', 'health')
|
|
// ->label('sdk.method', 'getStats')
|
|
->label('docs', false)
|
|
->inject('response')
|
|
->inject('register')
|
|
->action(function ($response, $register) {
|
|
/** @var Appwrite\Utopia\Response $response */
|
|
/** @var Utopia\Registry\Registry $register */
|
|
|
|
$device = Storage::getDevice('files');
|
|
$cache = $register->get('cache');
|
|
|
|
$cacheStats = $cache->info();
|
|
|
|
$response
|
|
->json([
|
|
'storage' => [
|
|
'used' => Storage::human($device->getDirectorySize($device->getRoot().'/')),
|
|
'partitionTotal' => Storage::human($device->getPartitionTotalSpace()),
|
|
'partitionFree' => Storage::human($device->getPartitionFreeSpace()),
|
|
],
|
|
'cache' => [
|
|
'uptime' => $cacheStats['uptime_in_seconds'] ?? 0,
|
|
'clients' => $cacheStats['connected_clients'] ?? 0,
|
|
'hits' => $cacheStats['keyspace_hits'] ?? 0,
|
|
'misses' => $cacheStats['keyspace_misses'] ?? 0,
|
|
'memory_used' => $cacheStats['used_memory'] ?? 0,
|
|
'memory_used_human' => $cacheStats['used_memory_human'] ?? 0,
|
|
'memory_used_peak' => $cacheStats['used_memory_peak'] ?? 0,
|
|
'memory_used_peak_human' => $cacheStats['used_memory_peak_human'] ?? 0,
|
|
],
|
|
]);
|
|
});
|