From 886d01405f5da60c62e4fedd7d0c8d212caf1202 Mon Sep 17 00:00:00 2001
From: ShadowArcanist <162910371+ShadowArcanist@users.noreply.github.com>
Date: Sat, 28 Mar 2026 17:48:10 +0530
Subject: [PATCH] feat(applications): add configurable restart loop limit
---
app/Actions/Docker/GetContainersStatus.php | 17 +--
app/Livewire/Project/Application/Advanced.php | 19 +++
app/Models/Application.php | 1 +
.../Application/RestartLimitReached.php | 140 ++++++++++++++++++
...rt_count_to_applications_and_databases.php | 22 +++
...pplication-restart-limit-reached.blade.php | 7 +
.../project/application/advanced.blade.php | 8 +
7 files changed, 204 insertions(+), 10 deletions(-)
create mode 100644 app/Notifications/Application/RestartLimitReached.php
create mode 100644 database/migrations/2026_03_27_000000_add_max_restart_count_to_applications_and_databases.php
create mode 100644 resources/views/emails/application-restart-limit-reached.blade.php
diff --git a/app/Actions/Docker/GetContainersStatus.php b/app/Actions/Docker/GetContainersStatus.php
index 5966876c6..cfac83583 100644
--- a/app/Actions/Docker/GetContainersStatus.php
+++ b/app/Actions/Docker/GetContainersStatus.php
@@ -2,6 +2,7 @@
namespace App\Actions\Docker;
+use App\Actions\Application\StopApplication;
use App\Actions\Database\StartDatabaseProxy;
use App\Actions\Database\StopDatabaseProxy;
use App\Actions\Shared\ComplexStatusCheck;
@@ -9,6 +10,7 @@ use App\Events\ServiceChecked;
use App\Models\ApplicationPreview;
use App\Models\Server;
use App\Models\ServiceDatabase;
+use App\Notifications\Application\RestartLimitReached as ApplicationRestartLimitReached;
use App\Services\ContainerStatusAggregator;
use App\Traits\CalculatesExcludedStatus;
use Illuminate\Support\Arr;
@@ -475,16 +477,11 @@ class GetContainersStatus
'last_restart_type' => 'crash',
]);
- // Send notification
- $containerName = $application->name;
- $projectUuid = data_get($application, 'environment.project.uuid');
- $environmentName = data_get($application, 'environment.name');
- $applicationUuid = data_get($application, 'uuid');
-
- if ($projectUuid && $applicationUuid && $environmentName) {
- $url = base_url().'/project/'.$projectUuid.'/'.$environmentName.'/application/'.$applicationUuid;
- } else {
- $url = null;
+ // Check if restart limit has been reached
+ $maxAllowedRestarts = $application->max_restart_count ?? 0;
+ if ($maxAllowedRestarts > 0 && $maxRestartCount >= $maxAllowedRestarts && $previousRestartCount < $maxAllowedRestarts) {
+ StopApplication::dispatch($application);
+ $application->environment->project->team?->notify(new ApplicationRestartLimitReached($application));
}
}
diff --git a/app/Livewire/Project/Application/Advanced.php b/app/Livewire/Project/Application/Advanced.php
index cf7ef3e0b..9954c3422 100644
--- a/app/Livewire/Project/Application/Advanced.php
+++ b/app/Livewire/Project/Application/Advanced.php
@@ -82,6 +82,9 @@ class Advanced extends Component
#[Validate(['boolean'])]
public bool $isConnectToDockerNetworkEnabled = false;
+ #[Validate(['integer', 'min:0'])]
+ public int $maxRestartCount = 10;
+
public function mount()
{
try {
@@ -144,6 +147,7 @@ class Advanced extends Component
$this->disableBuildCache = $this->application->settings->disable_build_cache;
$this->injectBuildArgsToDockerfile = $this->application->settings->inject_build_args_to_dockerfile ?? true;
$this->includeSourceCommitInBuild = $this->application->settings->include_source_commit_in_build ?? false;
+ $this->maxRestartCount = $this->application->max_restart_count ?? 10;
}
}
@@ -252,6 +256,21 @@ class Advanced extends Component
}
}
+ public function saveMaxRestartCount()
+ {
+ try {
+ $this->authorize('update', $this->application);
+ $this->validate([
+ 'maxRestartCount' => 'integer|min:0',
+ ]);
+ $this->application->max_restart_count = $this->maxRestartCount;
+ $this->application->save();
+ $this->dispatch('success', 'Max restart count saved.');
+ } catch (\Throwable $e) {
+ return handleError($e, $this);
+ }
+ }
+
public function render()
{
return view('livewire.project.application.advanced');
diff --git a/app/Models/Application.php b/app/Models/Application.php
index 4cc2dcf74..85a04041b 100644
--- a/app/Models/Application.php
+++ b/app/Models/Application.php
@@ -125,6 +125,7 @@ class Application extends BaseModel
protected $casts = [
'http_basic_auth_password' => 'encrypted',
'restart_count' => 'integer',
+ 'max_restart_count' => 'integer',
'last_restart_at' => 'datetime',
];
diff --git a/app/Notifications/Application/RestartLimitReached.php b/app/Notifications/Application/RestartLimitReached.php
new file mode 100644
index 000000000..8709e7cd3
--- /dev/null
+++ b/app/Notifications/Application/RestartLimitReached.php
@@ -0,0 +1,140 @@
+onQueue('high');
+ $this->resource_name = data_get($resource, 'name');
+ $this->project_uuid = data_get($resource, 'environment.project.uuid');
+ $this->environment_uuid = data_get($resource, 'environment.uuid');
+ $this->environment_name = data_get($resource, 'environment.name');
+ $this->fqdn = data_get($resource, 'fqdn', null);
+ $this->restart_count = $resource->restart_count;
+ $this->max_restart_count = $resource->max_restart_count;
+ if (str($this->fqdn)->explode(',')->count() > 1) {
+ $this->fqdn = str($this->fqdn)->explode(',')->first();
+ }
+ $this->resource_url = base_url()."/project/{$this->project_uuid}/environment/{$this->environment_uuid}/application/{$this->resource->uuid}";
+ }
+
+ public function via(object $notifiable): array
+ {
+ return $notifiable->getEnabledChannels('status_change');
+ }
+
+ public function toMail(): MailMessage
+ {
+ $mail = new MailMessage;
+ $mail->subject("Coolify: {$this->resource_name} stopped - restart limit reached ({$this->restart_count}/{$this->max_restart_count})");
+ $mail->view('emails.application-restart-limit-reached', [
+ 'name' => $this->resource_name,
+ 'fqdn' => $this->fqdn,
+ 'resource_url' => $this->resource_url,
+ 'restart_count' => $this->restart_count,
+ 'max_restart_count' => $this->max_restart_count,
+ ]);
+
+ return $mail;
+ }
+
+ public function toDiscord(): DiscordMessage
+ {
+ return new DiscordMessage(
+ title: ':warning: Restart limit reached',
+ description: "{$this->resource_name} has been stopped after {$this->restart_count} restarts (limit: {$this->max_restart_count}).\n\n[Open Application in Coolify]({$this->resource_url})",
+ color: DiscordMessage::errorColor(),
+ isCritical: true,
+ );
+ }
+
+ public function toTelegram(): array
+ {
+ $message = "Coolify: {$this->resource_name} has been stopped after {$this->restart_count} restarts (limit: {$this->max_restart_count}).";
+
+ return [
+ 'message' => $message,
+ 'buttons' => [
+ [
+ 'text' => 'Open Application in Coolify',
+ 'url' => $this->resource_url,
+ ],
+ ],
+ ];
+ }
+
+ public function toPushover(): PushoverMessage
+ {
+ $message = "{$this->resource_name} has been stopped after {$this->restart_count} restarts (limit: {$this->max_restart_count}).";
+
+ return new PushoverMessage(
+ title: 'Restart limit reached',
+ level: 'error',
+ message: $message,
+ buttons: [
+ [
+ 'text' => 'Open Application in Coolify',
+ 'url' => $this->resource_url,
+ ],
+ ],
+ );
+ }
+
+ public function toSlack(): SlackMessage
+ {
+ $title = 'Restart limit reached';
+ $description = "{$this->resource_name} has been stopped after {$this->restart_count} restarts (limit: {$this->max_restart_count})";
+
+ $description .= "\n\n*Project:* ".data_get($this->resource, 'environment.project.name');
+ $description .= "\n*Environment:* {$this->environment_name}";
+ $description .= "\n*Application URL:* {$this->resource_url}";
+
+ return new SlackMessage(
+ title: $title,
+ description: $description,
+ color: SlackMessage::errorColor()
+ );
+ }
+
+ public function toWebhook(): array
+ {
+ return [
+ 'success' => false,
+ 'message' => 'Restart limit reached',
+ 'event' => 'restart_limit_reached',
+ 'application_name' => $this->resource_name,
+ 'application_uuid' => $this->resource->uuid,
+ 'restart_count' => $this->restart_count,
+ 'max_restart_count' => $this->max_restart_count,
+ 'url' => $this->resource_url,
+ 'project' => data_get($this->resource, 'environment.project.name'),
+ 'environment' => $this->environment_name,
+ 'fqdn' => $this->fqdn,
+ ];
+ }
+}
diff --git a/database/migrations/2026_03_27_000000_add_max_restart_count_to_applications_and_databases.php b/database/migrations/2026_03_27_000000_add_max_restart_count_to_applications_and_databases.php
new file mode 100644
index 000000000..578959c9a
--- /dev/null
+++ b/database/migrations/2026_03_27_000000_add_max_restart_count_to_applications_and_databases.php
@@ -0,0 +1,22 @@
+integer('max_restart_count')->default(10)->after('restart_count');
+ });
+ }
+
+ public function down(): void
+ {
+ Schema::table('applications', function (Blueprint $blueprint) {
+ $blueprint->dropColumn('max_restart_count');
+ });
+ }
+};
diff --git a/resources/views/emails/application-restart-limit-reached.blade.php b/resources/views/emails/application-restart-limit-reached.blade.php
new file mode 100644
index 000000000..e5fa01c00
--- /dev/null
+++ b/resources/views/emails/application-restart-limit-reached.blade.php
@@ -0,0 +1,7 @@
+
Starting a docker compose based resource will have an internal network.
If you connect to a Coolify defined network, you maybe need to use different internal DNS names to connect to a resource.
For more information, check this."
canGate="update" :canResource="$application" />
@endif
+