2023-11-16 23:37:09 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace App\Actions\Server;
|
|
|
|
|
|
|
|
use Lorisleiva\Actions\Concerns\AsAction;
|
|
|
|
use App\Models\Server;
|
|
|
|
|
|
|
|
class InstallLogDrain
|
|
|
|
{
|
|
|
|
use AsAction;
|
2023-11-17 20:16:25 +00:00
|
|
|
public function handle(Server $server)
|
2023-11-16 23:37:09 +00:00
|
|
|
{
|
2023-11-17 20:16:25 +00:00
|
|
|
if ($server->settings->is_logdrain_newrelic_enabled) {
|
|
|
|
$type = 'newrelic';
|
|
|
|
} else if ($server->settings->is_logdrain_highlight_enabled) {
|
|
|
|
$type = 'highlight';
|
|
|
|
} else if ($server->settings->is_logdrain_axiom_enabled) {
|
|
|
|
$type = 'axiom';
|
2023-12-01 10:13:58 +00:00
|
|
|
} else if ($server->settings->is_logdrain_custom_enabled) {
|
|
|
|
$type = 'custom';
|
2023-11-17 20:16:25 +00:00
|
|
|
} else {
|
|
|
|
$type = 'none';
|
|
|
|
}
|
2023-11-17 09:21:19 +00:00
|
|
|
try {
|
|
|
|
if ($type === 'none') {
|
|
|
|
$command = [
|
|
|
|
"echo 'Stopping old Fluent Bit'",
|
|
|
|
"docker rm -f coolify-log-drain || true",
|
|
|
|
];
|
|
|
|
return instant_remote_process($command, $server);
|
|
|
|
} else if ($type === 'newrelic') {
|
|
|
|
if (!$server->settings->is_logdrain_newrelic_enabled) {
|
|
|
|
throw new \Exception('New Relic log drain is not enabled.');
|
|
|
|
}
|
|
|
|
$config = base64_encode("
|
2023-11-16 23:37:09 +00:00
|
|
|
[SERVICE]
|
|
|
|
Flush 5
|
|
|
|
Daemon off
|
|
|
|
Tag container_logs
|
2023-11-17 09:50:02 +00:00
|
|
|
Log_Level debug
|
|
|
|
Parsers_File parsers.conf
|
2023-11-16 23:37:09 +00:00
|
|
|
[INPUT]
|
|
|
|
Name forward
|
|
|
|
Buffer_Chunk_Size 1M
|
|
|
|
Buffer_Max_Size 6M
|
|
|
|
[FILTER]
|
|
|
|
Name grep
|
|
|
|
Match *
|
|
|
|
Exclude log 127.0.0.1
|
|
|
|
[FILTER]
|
|
|
|
Name modify
|
|
|
|
Match *
|
|
|
|
Set server_name {$server->name}
|
|
|
|
[OUTPUT]
|
|
|
|
Name nrlogs
|
|
|
|
Match *
|
|
|
|
license_key \${LICENSE_KEY}
|
|
|
|
# https://log-api.eu.newrelic.com/log/v1 - EU
|
|
|
|
# https://log-api.newrelic.com/log/v1 - US
|
|
|
|
base_uri \${BASE_URI}
|
|
|
|
");
|
2023-11-17 09:21:19 +00:00
|
|
|
} else if ($type === 'highlight') {
|
|
|
|
if (!$server->settings->is_logdrain_highlight_enabled) {
|
|
|
|
throw new \Exception('Highlight log drain is not enabled.');
|
|
|
|
}
|
|
|
|
$config = base64_encode("
|
|
|
|
[SERVICE]
|
2023-11-17 09:50:02 +00:00
|
|
|
Flush 5
|
2023-11-17 09:21:19 +00:00
|
|
|
Daemon off
|
|
|
|
Log_Level debug
|
2023-11-17 09:50:02 +00:00
|
|
|
Parsers_File parsers.conf
|
2023-11-17 09:21:19 +00:00
|
|
|
[INPUT]
|
|
|
|
Name forward
|
|
|
|
tag \${HIGHLIGHT_PROJECT_ID}
|
|
|
|
Buffer_Chunk_Size 1M
|
|
|
|
Buffer_Max_Size 6M
|
|
|
|
[OUTPUT]
|
|
|
|
Name forward
|
|
|
|
Match *
|
|
|
|
Host otel.highlight.io
|
|
|
|
Port 24224
|
|
|
|
");
|
|
|
|
} else if ($type === 'axiom') {
|
|
|
|
if (!$server->settings->is_logdrain_axiom_enabled) {
|
|
|
|
throw new \Exception('Axiom log drain is not enabled.');
|
|
|
|
}
|
|
|
|
$config = base64_encode("
|
2023-11-16 23:37:09 +00:00
|
|
|
[SERVICE]
|
|
|
|
Flush 5
|
|
|
|
Daemon off
|
2023-11-17 09:50:02 +00:00
|
|
|
Log_Level debug
|
|
|
|
Parsers_File parsers.conf
|
2023-11-16 23:37:09 +00:00
|
|
|
[INPUT]
|
|
|
|
Name forward
|
|
|
|
Buffer_Chunk_Size 1M
|
|
|
|
Buffer_Max_Size 6M
|
|
|
|
[FILTER]
|
|
|
|
Name grep
|
|
|
|
Match *
|
|
|
|
Exclude log 127.0.0.1
|
|
|
|
[FILTER]
|
|
|
|
Name modify
|
|
|
|
Match *
|
|
|
|
Set server_name {$server->name}
|
|
|
|
[OUTPUT]
|
2023-11-17 09:21:19 +00:00
|
|
|
Name http
|
|
|
|
Match *
|
|
|
|
Host api.axiom.co
|
|
|
|
Port 443
|
|
|
|
URI /v1/datasets/\${AXIOM_DATASET_NAME}/ingest
|
|
|
|
# Authorization Bearer should be an API token
|
|
|
|
Header Authorization Bearer \${AXIOM_API_KEY}
|
|
|
|
compress gzip
|
|
|
|
format json
|
|
|
|
json_date_key _time
|
|
|
|
json_date_format iso8601
|
|
|
|
tls On
|
|
|
|
");
|
2023-12-01 10:13:58 +00:00
|
|
|
} else if ($type === 'custom') {
|
|
|
|
if (!$server->settings->is_logdrain_custom_enabled) {
|
|
|
|
throw new \Exception('Custom log drain is not enabled.');
|
|
|
|
}
|
|
|
|
$config = base64_encode($server->settings->logdrain_custom_config);
|
|
|
|
$parsers = base64_encode($server->settings->logdrain_custom_config_parser);
|
2023-11-17 09:21:19 +00:00
|
|
|
} else {
|
|
|
|
throw new \Exception('Unknown log drain type.');
|
|
|
|
}
|
2023-12-01 10:13:58 +00:00
|
|
|
if ($type !== 'custom') {
|
|
|
|
$parsers = base64_encode("
|
2023-11-17 09:50:02 +00:00
|
|
|
[PARSER]
|
2023-12-01 10:13:58 +00:00
|
|
|
Name empty_line_skipper
|
|
|
|
Format regex
|
|
|
|
Regex /^(?!\s*$).+/
|
2023-11-17 09:50:02 +00:00
|
|
|
");
|
2023-12-01 10:13:58 +00:00
|
|
|
}
|
2023-11-17 09:21:19 +00:00
|
|
|
$compose = base64_encode("
|
2023-11-16 23:37:09 +00:00
|
|
|
services:
|
|
|
|
coolify-log-drain:
|
|
|
|
image: cr.fluentbit.io/fluent/fluent-bit:2.0
|
|
|
|
container_name: coolify-log-drain
|
|
|
|
command: -c /fluent-bit.conf
|
|
|
|
env_file:
|
|
|
|
- .env
|
|
|
|
volumes:
|
|
|
|
- ./fluent-bit.conf:/fluent-bit.conf
|
2023-11-17 09:50:02 +00:00
|
|
|
- ./parsers.conf:/parsers.conf
|
2023-11-16 23:37:09 +00:00
|
|
|
ports:
|
|
|
|
- 127.0.0.1:24224:24224
|
2023-11-17 19:08:21 +00:00
|
|
|
restart: unless-stopped
|
2023-11-16 23:37:09 +00:00
|
|
|
");
|
2023-11-17 09:21:19 +00:00
|
|
|
$readme = base64_encode('# New Relic Log Drain
|
2023-11-16 23:37:09 +00:00
|
|
|
This log drain is based on [Fluent Bit](https://fluentbit.io/) and New Relic Log Forwarder.
|
|
|
|
|
|
|
|
Files:
|
|
|
|
- `fluent-bit.conf` - configuration file for Fluent Bit
|
|
|
|
- `docker-compose.yml` - docker-compose file to run Fluent Bit
|
|
|
|
- `.env` - environment variables for Fluent Bit
|
|
|
|
');
|
2023-11-17 09:21:19 +00:00
|
|
|
$license_key = $server->settings->logdrain_newrelic_license_key;
|
|
|
|
$base_uri = $server->settings->logdrain_newrelic_base_uri;
|
|
|
|
$base_path = config('coolify.base_config_path');
|
2023-11-16 23:37:09 +00:00
|
|
|
|
2023-11-17 09:21:19 +00:00
|
|
|
$config_path = $base_path . '/log-drains';
|
|
|
|
$fluent_bit_config = $config_path . '/fluent-bit.conf';
|
2023-11-17 09:50:02 +00:00
|
|
|
$parsers_config = $config_path . '/parsers.conf';
|
2023-11-17 09:21:19 +00:00
|
|
|
$compose_path = $config_path . '/docker-compose.yml';
|
|
|
|
$readme_path = $config_path . '/README.md';
|
|
|
|
$command = [
|
|
|
|
"echo 'Saving configuration'",
|
|
|
|
"mkdir -p $config_path",
|
2023-11-17 09:50:02 +00:00
|
|
|
"echo '{$parsers}' | base64 -d > $parsers_config",
|
2023-11-17 09:21:19 +00:00
|
|
|
"echo '{$config}' | base64 -d > $fluent_bit_config",
|
|
|
|
"echo '{$compose}' | base64 -d > $compose_path",
|
|
|
|
"echo '{$readme}' | base64 -d > $readme_path",
|
|
|
|
"test -f $config_path/.env && rm $config_path/.env",
|
2023-11-16 23:37:09 +00:00
|
|
|
|
|
|
|
];
|
2023-11-17 09:21:19 +00:00
|
|
|
if ($type === 'newrelic') {
|
|
|
|
$add_envs_command = [
|
|
|
|
"echo LICENSE_KEY=$license_key >> $config_path/.env",
|
|
|
|
"echo BASE_URI=$base_uri >> $config_path/.env",
|
|
|
|
];
|
|
|
|
} else if ($type === 'highlight') {
|
|
|
|
$add_envs_command = [
|
|
|
|
"echo HIGHLIGHT_PROJECT_ID={$server->settings->logdrain_highlight_project_id} >> $config_path/.env",
|
|
|
|
];
|
|
|
|
} else if ($type === 'axiom') {
|
|
|
|
$add_envs_command = [
|
|
|
|
"echo AXIOM_DATASET_NAME={$server->settings->logdrain_axiom_dataset_name} >> $config_path/.env",
|
|
|
|
"echo AXIOM_API_KEY={$server->settings->logdrain_axiom_api_key} >> $config_path/.env",
|
|
|
|
];
|
2023-12-01 10:13:58 +00:00
|
|
|
} else if ($type === 'custom') {
|
|
|
|
$add_envs_command = [
|
|
|
|
"touch $config_path/.env"
|
|
|
|
];
|
|
|
|
} else {
|
|
|
|
throw new \Exception('Unknown log drain type.');
|
2023-11-17 09:21:19 +00:00
|
|
|
}
|
|
|
|
$restart_command = [
|
|
|
|
"echo 'Stopping old Fluent Bit'",
|
|
|
|
"cd $config_path && docker rm -f coolify-log-drain || true",
|
|
|
|
"echo 'Starting Fluent Bit'",
|
|
|
|
"cd $config_path && docker compose up -d --remove-orphans",
|
2023-11-16 23:37:09 +00:00
|
|
|
];
|
2023-11-17 09:21:19 +00:00
|
|
|
$command = array_merge($command, $add_envs_command, $restart_command);
|
|
|
|
return instant_remote_process($command, $server);
|
|
|
|
} catch (\Throwable $e) {
|
|
|
|
return handleError($e);
|
2023-11-16 23:37:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|