2023-03-24 14:47:58 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace App\Models;
|
|
|
|
|
2023-11-22 14:18:49 +00:00
|
|
|
use App\Enums\ApplicationDeploymentStatus;
|
2023-09-21 15:48:31 +00:00
|
|
|
use App\Enums\ProxyStatus;
|
|
|
|
use App\Enums\ProxyTypes;
|
2023-11-16 10:53:37 +00:00
|
|
|
use App\Notifications\Server\Revived;
|
|
|
|
use App\Notifications\Server\Unreachable;
|
2023-05-03 05:23:45 +00:00
|
|
|
use Illuminate\Database\Eloquent\Builder;
|
2023-10-04 08:57:44 +00:00
|
|
|
use Illuminate\Database\Eloquent\Casts\Attribute;
|
2023-11-22 14:18:49 +00:00
|
|
|
use Illuminate\Support\Carbon;
|
|
|
|
use Illuminate\Support\Collection;
|
|
|
|
use Illuminate\Support\Facades\Process;
|
2023-05-03 05:23:45 +00:00
|
|
|
use Spatie\SchemalessAttributes\Casts\SchemalessAttributes;
|
2023-06-20 18:19:31 +00:00
|
|
|
use Spatie\SchemalessAttributes\SchemalessAttributesTrait;
|
2023-10-06 22:51:01 +00:00
|
|
|
use Illuminate\Support\Str;
|
2023-11-28 13:05:55 +00:00
|
|
|
use Illuminate\Support\Stringable;
|
2023-05-03 05:23:45 +00:00
|
|
|
|
2023-03-24 14:47:58 +00:00
|
|
|
class Server extends BaseModel
|
|
|
|
{
|
2023-06-20 18:19:31 +00:00
|
|
|
use SchemalessAttributesTrait;
|
2023-11-22 14:18:49 +00:00
|
|
|
public static $batch_counter = 0;
|
2023-08-08 09:51:36 +00:00
|
|
|
|
2023-08-23 08:14:39 +00:00
|
|
|
protected static function booted()
|
|
|
|
{
|
2023-10-09 09:00:18 +00:00
|
|
|
static::saving(function ($server) {
|
2023-10-09 18:12:03 +00:00
|
|
|
$payload = [];
|
|
|
|
if ($server->user) {
|
|
|
|
$payload['user'] = Str::of($server->user)->trim();
|
|
|
|
}
|
|
|
|
if ($server->ip) {
|
|
|
|
$payload['ip'] = Str::of($server->ip)->trim();
|
|
|
|
}
|
|
|
|
$server->forceFill($payload);
|
2023-10-06 22:51:01 +00:00
|
|
|
});
|
2023-08-23 08:14:39 +00:00
|
|
|
static::created(function ($server) {
|
|
|
|
ServerSetting::create([
|
|
|
|
'server_id' => $server->id,
|
|
|
|
]);
|
|
|
|
});
|
|
|
|
static::deleting(function ($server) {
|
2023-08-29 13:51:30 +00:00
|
|
|
$server->destinations()->each(function ($destination) {
|
|
|
|
$destination->delete();
|
|
|
|
});
|
2023-08-23 08:14:39 +00:00
|
|
|
$server->settings()->delete();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-08-08 09:51:36 +00:00
|
|
|
public $casts = [
|
|
|
|
'proxy' => SchemalessAttributes::class,
|
2023-11-17 09:21:19 +00:00
|
|
|
'logdrain_axiom_api_key' => 'encrypted',
|
|
|
|
'logdrain_newrelic_license_key' => 'encrypted',
|
2023-08-08 09:51:36 +00:00
|
|
|
];
|
2023-06-20 18:19:31 +00:00
|
|
|
protected $schemalessAttributes = [
|
|
|
|
'proxy',
|
|
|
|
];
|
2023-08-23 08:14:39 +00:00
|
|
|
protected $guarded = [];
|
2023-08-08 09:51:36 +00:00
|
|
|
|
|
|
|
static public function isReachable()
|
|
|
|
{
|
|
|
|
return Server::ownedByCurrentTeam()->whereRelation('settings', 'is_reachable', true);
|
|
|
|
}
|
|
|
|
|
|
|
|
static public function ownedByCurrentTeam(array $select = ['*'])
|
|
|
|
{
|
2023-08-22 15:44:49 +00:00
|
|
|
$teamId = currentTeam()->id;
|
2023-08-08 09:51:36 +00:00
|
|
|
$selectArray = collect($select)->concat(['id']);
|
2023-12-04 14:08:24 +00:00
|
|
|
return Server::whereTeamId($teamId)->with('settings', 'swarmDockers', 'standaloneDockers')->select($selectArray->all())->orderBy('name');
|
2023-08-08 09:51:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static public function isUsable()
|
|
|
|
{
|
2024-01-16 14:19:14 +00:00
|
|
|
return Server::ownedByCurrentTeam()->whereRelation('settings', 'is_reachable', true)->whereRelation('settings', 'is_usable', true)->whereRelation('settings', 'is_swarm_worker', false)->whereRelation('settings', 'is_build_server', false);
|
2023-08-08 09:51:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static public function destinationsByServer(string $server_id)
|
|
|
|
{
|
|
|
|
$server = Server::ownedByCurrentTeam()->get()->where('id', $server_id)->firstOrFail();
|
|
|
|
$standaloneDocker = collect($server->standaloneDockers->all());
|
|
|
|
$swarmDocker = collect($server->swarmDockers->all());
|
|
|
|
return $standaloneDocker->concat($swarmDocker);
|
|
|
|
}
|
|
|
|
public function settings()
|
|
|
|
{
|
|
|
|
return $this->hasOne(ServerSetting::class);
|
|
|
|
}
|
2023-12-04 14:08:24 +00:00
|
|
|
public function addInitialNetwork()
|
|
|
|
{
|
2023-11-28 14:49:24 +00:00
|
|
|
if ($this->id === 0) {
|
|
|
|
if ($this->isSwarm()) {
|
|
|
|
SwarmDocker::create([
|
|
|
|
'id' => 0,
|
|
|
|
'name' => 'coolify',
|
|
|
|
'network' => 'coolify-overlay',
|
|
|
|
'server_id' => $this->id,
|
|
|
|
]);
|
|
|
|
} else {
|
|
|
|
StandaloneDocker::create([
|
|
|
|
'id' => 0,
|
|
|
|
'name' => 'coolify',
|
|
|
|
'network' => 'coolify',
|
|
|
|
'server_id' => $this->id,
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if ($this->isSwarm()) {
|
|
|
|
SwarmDocker::create([
|
2023-11-28 17:31:04 +00:00
|
|
|
'name' => 'coolify-overlay',
|
2023-11-28 14:49:24 +00:00
|
|
|
'network' => 'coolify-overlay',
|
|
|
|
'server_id' => $this->id,
|
|
|
|
]);
|
|
|
|
} else {
|
|
|
|
StandaloneDocker::create([
|
2024-01-17 13:11:46 +00:00
|
|
|
'name' => 'coolify',
|
2023-11-28 14:49:24 +00:00
|
|
|
'network' => 'coolify',
|
|
|
|
'server_id' => $this->id,
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-09-25 07:17:42 +00:00
|
|
|
public function proxyType()
|
|
|
|
{
|
2023-10-12 06:51:32 +00:00
|
|
|
$proxyType = $this->proxy->get('type');
|
|
|
|
if ($proxyType === ProxyTypes::NONE->value) {
|
|
|
|
return $proxyType;
|
|
|
|
}
|
|
|
|
if (is_null($proxyType)) {
|
2023-09-21 15:48:31 +00:00
|
|
|
$this->proxy->type = ProxyTypes::TRAEFIK_V2->value;
|
|
|
|
$this->proxy->status = ProxyStatus::EXITED->value;
|
|
|
|
$this->save();
|
|
|
|
}
|
2023-09-20 13:42:41 +00:00
|
|
|
return $this->proxy->get('type');
|
|
|
|
}
|
2023-06-20 18:19:31 +00:00
|
|
|
public function scopeWithProxy(): Builder
|
2023-05-30 13:52:17 +00:00
|
|
|
{
|
2023-06-20 18:19:31 +00:00
|
|
|
return $this->proxy->modelScope();
|
2023-05-30 13:52:17 +00:00
|
|
|
}
|
2023-08-08 09:51:36 +00:00
|
|
|
|
2023-11-16 13:29:01 +00:00
|
|
|
public function isLocalhost()
|
|
|
|
{
|
|
|
|
return $this->ip === 'host.docker.internal' || $this->id === 0;
|
2023-11-16 13:28:26 +00:00
|
|
|
}
|
2024-01-17 10:52:56 +00:00
|
|
|
static public function buildServers($teamId)
|
|
|
|
{
|
|
|
|
return Server::whereTeamId($teamId)->whereRelation('settings', 'is_reachable', true)->whereRelation('settings', 'is_build_server', true);
|
|
|
|
}
|
2023-11-17 11:47:15 +00:00
|
|
|
public function skipServer()
|
|
|
|
{
|
|
|
|
if ($this->ip === '1.2.3.4') {
|
|
|
|
ray('skipping 1.2.3.4');
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2023-12-19 14:36:59 +00:00
|
|
|
public function isServerReady(int $tries = 3)
|
2023-11-16 10:53:37 +00:00
|
|
|
{
|
2023-12-20 10:21:17 +00:00
|
|
|
if ($this->skipServer()) {
|
|
|
|
return false;
|
|
|
|
}
|
2023-12-21 08:57:39 +00:00
|
|
|
$serverUptimeCheckNumber = $this->unreachable_count;
|
2023-12-20 10:21:17 +00:00
|
|
|
if ($this->unreachable_count < $tries) {
|
|
|
|
$serverUptimeCheckNumber = $this->unreachable_count + 1;
|
|
|
|
}
|
2023-12-21 09:00:41 +00:00
|
|
|
if ($this->unreachable_count > $tries) {
|
|
|
|
$serverUptimeCheckNumber = $tries;
|
|
|
|
}
|
2023-12-21 08:57:39 +00:00
|
|
|
|
2023-12-19 14:36:59 +00:00
|
|
|
$serverUptimeCheckNumberMax = $tries;
|
2023-11-16 13:29:01 +00:00
|
|
|
|
2023-12-21 09:28:02 +00:00
|
|
|
// ray('server: ' . $this->name);
|
|
|
|
// ray('serverUptimeCheckNumber: ' . $serverUptimeCheckNumber);
|
|
|
|
// ray('serverUptimeCheckNumberMax: ' . $serverUptimeCheckNumberMax);
|
2023-11-17 13:46:04 +00:00
|
|
|
|
2023-12-19 14:16:08 +00:00
|
|
|
$result = $this->validateConnection();
|
|
|
|
if ($result) {
|
|
|
|
if ($this->unreachable_notification_sent === true) {
|
|
|
|
$this->update(['unreachable_notification_sent' => false]);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
} else {
|
2023-11-16 10:53:37 +00:00
|
|
|
if ($serverUptimeCheckNumber >= $serverUptimeCheckNumberMax) {
|
2023-12-19 14:16:08 +00:00
|
|
|
// Reached max number of retries
|
2023-11-16 10:53:37 +00:00
|
|
|
if ($this->unreachable_notification_sent === false) {
|
|
|
|
ray('Server unreachable, sending notification...');
|
2023-12-13 11:01:27 +00:00
|
|
|
$this->team?->notify(new Unreachable($this));
|
2023-11-16 10:53:37 +00:00
|
|
|
$this->update(['unreachable_notification_sent' => true]);
|
|
|
|
}
|
2023-12-19 14:16:08 +00:00
|
|
|
if ($this->settings->is_reachable === true) {
|
|
|
|
$this->settings()->update([
|
|
|
|
'is_reachable' => false,
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
2023-11-16 10:53:37 +00:00
|
|
|
foreach ($this->applications() as $application) {
|
|
|
|
$application->update(['status' => 'exited']);
|
|
|
|
}
|
|
|
|
foreach ($this->databases() as $database) {
|
|
|
|
$database->update(['status' => 'exited']);
|
|
|
|
}
|
2024-01-17 14:48:01 +00:00
|
|
|
foreach ($this->services()->get() as $service) {
|
2023-11-16 10:53:37 +00:00
|
|
|
$apps = $service->applications()->get();
|
|
|
|
$dbs = $service->databases()->get();
|
|
|
|
foreach ($apps as $app) {
|
|
|
|
$app->update(['status' => 'exited']);
|
|
|
|
}
|
|
|
|
foreach ($dbs as $db) {
|
|
|
|
$db->update(['status' => 'exited']);
|
|
|
|
}
|
|
|
|
}
|
2023-12-19 14:16:08 +00:00
|
|
|
} else {
|
2023-11-16 10:53:37 +00:00
|
|
|
$this->update([
|
2023-12-19 14:16:08 +00:00
|
|
|
'unreachable_count' => $this->unreachable_count + 1,
|
2023-11-16 10:53:37 +00:00
|
|
|
]);
|
|
|
|
}
|
2023-12-19 14:16:08 +00:00
|
|
|
return false;
|
2023-11-16 10:53:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
public function getDiskUsage()
|
|
|
|
{
|
|
|
|
return instant_remote_process(["df /| tail -1 | awk '{ print $5}' | sed 's/%//g'"], $this, false);
|
|
|
|
}
|
2023-11-28 12:12:25 +00:00
|
|
|
public function definedResources()
|
|
|
|
{
|
2023-11-27 08:39:43 +00:00
|
|
|
$applications = $this->applications();
|
|
|
|
$databases = $this->databases();
|
|
|
|
$services = $this->services();
|
|
|
|
return $applications->concat($databases)->concat($services->get());
|
|
|
|
}
|
2023-11-06 17:04:18 +00:00
|
|
|
public function hasDefinedResources()
|
2023-06-15 13:15:27 +00:00
|
|
|
{
|
2023-11-14 14:06:03 +00:00
|
|
|
$applications = $this->applications()->count() > 0;
|
|
|
|
$databases = $this->databases()->count() > 0;
|
|
|
|
$services = $this->services()->count() > 0;
|
2023-11-06 17:04:18 +00:00
|
|
|
if ($applications || $databases || $services) {
|
2023-06-15 13:15:27 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2023-08-08 09:51:36 +00:00
|
|
|
|
2023-09-11 20:29:34 +00:00
|
|
|
public function databases()
|
|
|
|
{
|
2023-08-29 13:51:30 +00:00
|
|
|
return $this->destinations()->map(function ($standaloneDocker) {
|
2023-10-24 12:31:28 +00:00
|
|
|
$postgresqls = data_get($standaloneDocker, 'postgresqls', collect([]));
|
|
|
|
$redis = data_get($standaloneDocker, 'redis', collect([]));
|
|
|
|
$mongodbs = data_get($standaloneDocker, 'mongodbs', collect([]));
|
|
|
|
$mysqls = data_get($standaloneDocker, 'mysqls', collect([]));
|
|
|
|
$mariadbs = data_get($standaloneDocker, 'mariadbs', collect([]));
|
|
|
|
return $postgresqls->concat($redis)->concat($mongodbs)->concat($mysqls)->concat($mariadbs);
|
2023-08-29 13:51:30 +00:00
|
|
|
})->flatten();
|
|
|
|
}
|
2023-06-15 13:15:27 +00:00
|
|
|
public function applications()
|
|
|
|
{
|
|
|
|
return $this->destinations()->map(function ($standaloneDocker) {
|
|
|
|
return $standaloneDocker->applications;
|
|
|
|
})->flatten();
|
|
|
|
}
|
2023-11-28 11:48:55 +00:00
|
|
|
public function dockerComposeBasedApplications()
|
|
|
|
{
|
|
|
|
return $this->applications()->filter(function ($application) {
|
|
|
|
return data_get($application, 'build_pack') === 'dockercompose';
|
|
|
|
});
|
|
|
|
}
|
|
|
|
public function dockerComposeBasedPreviewDeployments()
|
|
|
|
{
|
|
|
|
return $this->previews()->filter(function ($preview) {
|
|
|
|
$applicationId = data_get($preview, 'application_id');
|
|
|
|
$application = Application::find($applicationId);
|
|
|
|
if (!$application) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return data_get($application, 'build_pack') === 'dockercompose';
|
|
|
|
});
|
|
|
|
}
|
2023-09-25 07:17:42 +00:00
|
|
|
public function services()
|
|
|
|
{
|
2023-09-21 15:48:31 +00:00
|
|
|
return $this->hasMany(Service::class);
|
|
|
|
}
|
2023-10-04 08:57:44 +00:00
|
|
|
public function getIp(): Attribute
|
|
|
|
{
|
|
|
|
return Attribute::make(
|
|
|
|
get: function () {
|
|
|
|
if (isDev()) {
|
|
|
|
return '127.0.0.1';
|
|
|
|
}
|
2023-11-16 13:29:01 +00:00
|
|
|
if ($this->isLocalhost()) {
|
2023-10-04 08:57:44 +00:00
|
|
|
return base_ip();
|
|
|
|
}
|
|
|
|
return $this->ip;
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
2023-09-25 07:17:42 +00:00
|
|
|
public function previews()
|
|
|
|
{
|
2023-09-14 13:52:04 +00:00
|
|
|
return $this->destinations()->map(function ($standaloneDocker) {
|
|
|
|
return $standaloneDocker->applications->map(function ($application) {
|
|
|
|
return $application->previews;
|
|
|
|
})->flatten();
|
|
|
|
})->flatten();
|
|
|
|
}
|
|
|
|
|
2023-06-13 08:02:58 +00:00
|
|
|
public function destinations()
|
|
|
|
{
|
|
|
|
$standalone_docker = $this->hasMany(StandaloneDocker::class)->get();
|
|
|
|
$swarm_docker = $this->hasMany(SwarmDocker::class)->get();
|
|
|
|
return $standalone_docker->concat($swarm_docker);
|
|
|
|
}
|
2023-08-08 09:51:36 +00:00
|
|
|
|
2023-05-02 10:47:52 +00:00
|
|
|
public function standaloneDockers()
|
2023-04-25 12:43:35 +00:00
|
|
|
{
|
2023-05-02 10:47:52 +00:00
|
|
|
return $this->hasMany(StandaloneDocker::class);
|
|
|
|
}
|
2023-05-04 13:45:53 +00:00
|
|
|
|
2023-05-02 10:47:52 +00:00
|
|
|
public function swarmDockers()
|
|
|
|
{
|
|
|
|
return $this->hasMany(SwarmDocker::class);
|
2023-04-25 12:43:35 +00:00
|
|
|
}
|
2023-05-04 13:45:53 +00:00
|
|
|
|
2023-03-27 12:31:42 +00:00
|
|
|
public function privateKey()
|
2023-03-24 14:47:58 +00:00
|
|
|
{
|
2023-03-27 12:31:42 +00:00
|
|
|
return $this->belongsTo(PrivateKey::class);
|
2023-03-24 14:47:58 +00:00
|
|
|
}
|
2023-05-03 05:23:45 +00:00
|
|
|
|
2023-06-16 11:13:09 +00:00
|
|
|
public function muxFilename()
|
|
|
|
{
|
|
|
|
return "{$this->ip}_{$this->port}_{$this->user}";
|
|
|
|
}
|
2023-08-08 09:51:36 +00:00
|
|
|
|
2023-07-07 19:35:29 +00:00
|
|
|
public function team()
|
|
|
|
{
|
|
|
|
return $this->belongsTo(Team::class);
|
|
|
|
}
|
2023-09-11 20:29:34 +00:00
|
|
|
public function isProxyShouldRun()
|
|
|
|
{
|
2024-01-16 14:19:14 +00:00
|
|
|
if ($this->proxyType() === ProxyTypes::NONE->value || $this->settings->is_build_server) {
|
2023-09-25 07:17:42 +00:00
|
|
|
return false;
|
|
|
|
}
|
2023-10-17 13:40:47 +00:00
|
|
|
// foreach ($this->applications() as $application) {
|
|
|
|
// if (data_get($application, 'fqdn')) {
|
|
|
|
// $shouldRun = true;
|
|
|
|
// break;
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// ray($this->services()->get());
|
|
|
|
|
|
|
|
// if ($this->id === 0) {
|
|
|
|
// $settings = InstanceSettings::get();
|
|
|
|
// if (data_get($settings, 'fqdn')) {
|
|
|
|
// $shouldRun = true;
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
return true;
|
2023-09-11 20:29:34 +00:00
|
|
|
}
|
2023-09-25 07:17:42 +00:00
|
|
|
public function isFunctional()
|
|
|
|
{
|
2023-09-12 11:14:01 +00:00
|
|
|
return $this->settings->is_reachable && $this->settings->is_usable;
|
|
|
|
}
|
2023-11-17 19:08:21 +00:00
|
|
|
public function isLogDrainEnabled()
|
2023-11-17 11:22:45 +00:00
|
|
|
{
|
2023-12-01 10:13:58 +00:00
|
|
|
return $this->settings->is_logdrain_newrelic_enabled || $this->settings->is_logdrain_highlight_enabled || $this->settings->is_logdrain_axiom_enabled || $this->settings->is_logdrain_custom_enabled;
|
2023-11-17 10:13:16 +00:00
|
|
|
}
|
2023-11-28 12:21:32 +00:00
|
|
|
public function validateOS(): bool | Stringable
|
2023-11-21 10:39:19 +00:00
|
|
|
{
|
|
|
|
$os_release = instant_remote_process(['cat /etc/os-release'], $this);
|
2024-01-08 20:59:26 +00:00
|
|
|
$releaseLines = collect(explode("\n", $os_release));
|
2023-11-21 10:39:19 +00:00
|
|
|
$collectedData = collect([]);
|
2024-01-08 20:59:26 +00:00
|
|
|
foreach ($releaseLines as $line) {
|
|
|
|
$item = Str::of($line)->trim();
|
2023-11-21 10:39:19 +00:00
|
|
|
$collectedData->put($item->before('=')->value(), $item->after('=')->lower()->replace('"', '')->value());
|
|
|
|
}
|
|
|
|
$ID = data_get($collectedData, 'ID');
|
2023-11-28 12:12:25 +00:00
|
|
|
// $ID_LIKE = data_get($collectedData, 'ID_LIKE');
|
|
|
|
// $VERSION_ID = data_get($collectedData, 'VERSION_ID');
|
|
|
|
$supported = collect(SUPPORTED_OS)->filter(function ($supportedOs) use ($ID) {
|
|
|
|
if (str($supportedOs)->contains($ID)) {
|
|
|
|
return str($ID);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
if ($supported->count() === 1) {
|
2023-11-21 10:39:19 +00:00
|
|
|
ray('supported');
|
2023-11-28 12:17:59 +00:00
|
|
|
return str($supported->first());
|
2023-11-21 10:39:19 +00:00
|
|
|
} else {
|
|
|
|
ray('not supported');
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2023-11-28 14:49:24 +00:00
|
|
|
public function isSwarm()
|
|
|
|
{
|
2023-11-29 09:06:52 +00:00
|
|
|
return data_get($this, 'settings.is_swarm_manager') || data_get($this, 'settings.is_swarm_worker');
|
2023-11-28 14:49:24 +00:00
|
|
|
}
|
2023-12-18 13:01:25 +00:00
|
|
|
public function isSwarmManager()
|
|
|
|
{
|
|
|
|
return data_get($this, 'settings.is_swarm_manager');
|
|
|
|
}
|
|
|
|
public function isSwarmWorker()
|
|
|
|
{
|
|
|
|
return data_get($this, 'settings.is_swarm_worker');
|
|
|
|
}
|
2023-10-09 09:00:18 +00:00
|
|
|
public function validateConnection()
|
|
|
|
{
|
2024-01-26 07:54:56 +00:00
|
|
|
config()->set('coolify.mux_enabled', false);
|
|
|
|
|
2023-11-28 14:49:24 +00:00
|
|
|
$server = Server::find($this->id);
|
2023-12-19 11:24:43 +00:00
|
|
|
if (!$server) {
|
|
|
|
return false;
|
|
|
|
}
|
2023-12-04 19:47:32 +00:00
|
|
|
if ($server->skipServer()) {
|
2023-11-17 11:47:15 +00:00
|
|
|
return false;
|
|
|
|
}
|
2023-12-04 19:47:32 +00:00
|
|
|
$uptime = instant_remote_process(['uptime'], $server, false);
|
2023-10-09 09:00:18 +00:00
|
|
|
if (!$uptime) {
|
2023-12-04 19:47:32 +00:00
|
|
|
$server->settings()->update([
|
2023-11-17 11:22:45 +00:00
|
|
|
'is_reachable' => false,
|
|
|
|
]);
|
2023-10-09 09:00:18 +00:00
|
|
|
return false;
|
2023-11-21 10:39:19 +00:00
|
|
|
} else {
|
2023-12-04 19:47:32 +00:00
|
|
|
$server->settings()->update([
|
2023-11-21 10:39:19 +00:00
|
|
|
'is_reachable' => true,
|
|
|
|
]);
|
2023-11-28 14:49:24 +00:00
|
|
|
$server->update([
|
2023-11-21 10:39:19 +00:00
|
|
|
'unreachable_count' => 0,
|
|
|
|
]);
|
2023-10-09 09:00:18 +00:00
|
|
|
}
|
2023-11-17 11:22:45 +00:00
|
|
|
|
2023-12-04 19:47:32 +00:00
|
|
|
if (data_get($server, 'unreachable_notification_sent') === true) {
|
2023-12-13 11:01:27 +00:00
|
|
|
$server->team?->notify(new Revived($server));
|
2023-11-28 14:49:24 +00:00
|
|
|
$server->update(['unreachable_notification_sent' => false]);
|
2023-11-17 11:22:45 +00:00
|
|
|
}
|
2023-11-21 10:39:19 +00:00
|
|
|
|
2023-10-09 09:00:18 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
public function validateDockerEngine($throwError = false)
|
|
|
|
{
|
|
|
|
$dockerBinary = instant_remote_process(["command -v docker"], $this, false);
|
|
|
|
if (is_null($dockerBinary)) {
|
|
|
|
$this->settings->is_usable = false;
|
|
|
|
$this->settings->save();
|
|
|
|
if ($throwError) {
|
2023-11-21 10:39:19 +00:00
|
|
|
throw new \Exception('Server is not usable. Docker Engine is not installed.');
|
2023-10-09 09:00:18 +00:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
$this->settings->is_usable = true;
|
|
|
|
$this->settings->save();
|
2024-01-16 14:19:14 +00:00
|
|
|
$this->validateCoolifyNetwork(isSwarm: false, isBuildServer: $this->settings->is_build_server);
|
2023-11-28 14:49:24 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
public function validateDockerSwarm()
|
|
|
|
{
|
|
|
|
$swarmStatus = instant_remote_process(["docker info|grep -i swarm"], $this, false);
|
|
|
|
$swarmStatus = str($swarmStatus)->trim()->after(':')->trim();
|
|
|
|
if ($swarmStatus === 'inactive') {
|
|
|
|
throw new \Exception('Docker Swarm is not initiated. Please join the server to a swarm before continuing.');
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
$this->settings->is_usable = true;
|
|
|
|
$this->settings->save();
|
|
|
|
$this->validateCoolifyNetwork(isSwarm: true);
|
2023-10-09 09:00:18 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
public function validateDockerEngineVersion()
|
|
|
|
{
|
|
|
|
$dockerVersion = instant_remote_process(["docker version|head -2|grep -i version| awk '{print $2}'"], $this, false);
|
|
|
|
$dockerVersion = checkMinimumDockerEngineVersion($dockerVersion);
|
|
|
|
if (is_null($dockerVersion)) {
|
|
|
|
$this->settings->is_usable = false;
|
|
|
|
$this->settings->save();
|
|
|
|
return false;
|
|
|
|
}
|
2023-11-21 10:39:19 +00:00
|
|
|
$this->settings->is_reachable = true;
|
2023-10-09 09:00:18 +00:00
|
|
|
$this->settings->is_usable = true;
|
|
|
|
$this->settings->save();
|
|
|
|
return true;
|
|
|
|
}
|
2024-01-16 14:19:14 +00:00
|
|
|
public function validateCoolifyNetwork($isSwarm = false, $isBuildServer = false)
|
2023-10-24 12:31:28 +00:00
|
|
|
{
|
2024-01-16 14:19:14 +00:00
|
|
|
if ($isBuildServer) {
|
|
|
|
return;
|
|
|
|
}
|
2023-11-28 14:49:24 +00:00
|
|
|
if ($isSwarm) {
|
2023-11-28 17:31:04 +00:00
|
|
|
return instant_remote_process(["docker network create --attachable --driver overlay coolify-overlay >/dev/null 2>&1 || true"], $this, false);
|
2023-11-28 14:49:24 +00:00
|
|
|
} else {
|
|
|
|
return instant_remote_process(["docker network create coolify --attachable >/dev/null 2>&1 || true"], $this, false);
|
|
|
|
}
|
2023-10-09 09:00:18 +00:00
|
|
|
}
|
2023-11-24 14:48:23 +00:00
|
|
|
public function executeRemoteCommand(Collection $commands, ?ApplicationDeploymentQueue $loggingModel = null)
|
2023-11-22 14:18:49 +00:00
|
|
|
{
|
|
|
|
static::$batch_counter++;
|
|
|
|
foreach ($commands as $command) {
|
2023-11-23 20:02:30 +00:00
|
|
|
$realCommand = data_get($command, 'command');
|
|
|
|
if (is_null($realCommand)) {
|
|
|
|
throw new \RuntimeException('Command is not set');
|
2023-11-22 14:18:49 +00:00
|
|
|
}
|
|
|
|
$hidden = data_get($command, 'hidden', false);
|
|
|
|
$ignoreErrors = data_get($command, 'ignoreErrors', false);
|
|
|
|
$customOutputType = data_get($command, 'customOutputType');
|
2023-11-23 20:02:30 +00:00
|
|
|
$name = data_get($command, 'name');
|
|
|
|
$remoteCommand = generateSshCommand($this, $realCommand);
|
2023-11-22 14:18:49 +00:00
|
|
|
|
2023-11-23 20:02:30 +00:00
|
|
|
$process = Process::timeout(3600)->idleTimeout(3600)->start($remoteCommand, function (string $type, string $output) use ($realCommand, $hidden, $customOutputType, $loggingModel, $name) {
|
2023-11-22 14:18:49 +00:00
|
|
|
$output = str($output)->trim();
|
|
|
|
if ($output->startsWith('╔')) {
|
|
|
|
$output = "\n" . $output;
|
|
|
|
}
|
|
|
|
$newLogEntry = [
|
2023-11-23 20:02:30 +00:00
|
|
|
'command' => remove_iip($realCommand),
|
2023-11-22 14:18:49 +00:00
|
|
|
'output' => remove_iip($output),
|
|
|
|
'type' => $customOutputType ?? $type === 'err' ? 'stderr' : 'stdout',
|
|
|
|
'timestamp' => Carbon::now('UTC'),
|
|
|
|
'hidden' => $hidden,
|
|
|
|
'batch' => static::$batch_counter,
|
|
|
|
];
|
2023-11-24 14:48:23 +00:00
|
|
|
if ($loggingModel) {
|
|
|
|
if (!$loggingModel->logs) {
|
|
|
|
$newLogEntry['order'] = 1;
|
|
|
|
} else {
|
|
|
|
$previousLogs = json_decode($loggingModel->logs, associative: true, flags: JSON_THROW_ON_ERROR);
|
|
|
|
$newLogEntry['order'] = count($previousLogs) + 1;
|
|
|
|
}
|
|
|
|
if ($name) {
|
|
|
|
$newLogEntry['name'] = $name;
|
|
|
|
}
|
2023-11-23 20:02:30 +00:00
|
|
|
|
2023-11-24 14:48:23 +00:00
|
|
|
$previousLogs[] = $newLogEntry;
|
|
|
|
$loggingModel->logs = json_encode($previousLogs, flags: JSON_THROW_ON_ERROR);
|
|
|
|
$loggingModel->save();
|
|
|
|
}
|
2023-11-22 14:18:49 +00:00
|
|
|
});
|
2023-11-24 14:48:23 +00:00
|
|
|
if ($loggingModel) {
|
|
|
|
$loggingModel->update([
|
|
|
|
'current_process_id' => $process->id(),
|
|
|
|
]);
|
|
|
|
}
|
2023-11-22 14:18:49 +00:00
|
|
|
$processResult = $process->wait();
|
|
|
|
if ($processResult->exitCode() !== 0) {
|
|
|
|
if (!$ignoreErrors) {
|
2023-11-24 14:48:23 +00:00
|
|
|
if ($loggingModel) {
|
|
|
|
$status = ApplicationDeploymentStatus::FAILED->value;
|
|
|
|
$loggingModel->status = $status;
|
|
|
|
$loggingModel->save();
|
|
|
|
}
|
2023-11-22 14:18:49 +00:00
|
|
|
throw new \RuntimeException($processResult->errorOutput());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-11-23 20:02:30 +00:00
|
|
|
public function stopApplicationRelatedRunningContainers(string $applicationId, string $containerName)
|
|
|
|
{
|
|
|
|
$containers = getCurrentApplicationContainerStatus($this, $applicationId, 0);
|
|
|
|
$containers = $containers->filter(function ($container) use ($containerName) {
|
|
|
|
return data_get($container, 'Names') !== $containerName;
|
|
|
|
});
|
|
|
|
$containers->each(function ($container) {
|
|
|
|
$removableContainer = data_get($container, 'Names');
|
|
|
|
$this->server->executeRemoteCommand(
|
|
|
|
commands: collect([
|
|
|
|
'command' => "docker rm -f $removableContainer >/dev/null 2>&1",
|
|
|
|
'hidden' => true,
|
|
|
|
'ignoreErrors' => true
|
|
|
|
]),
|
|
|
|
loggingModel: $this->deploymentQueueEntry
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
2023-12-04 14:08:24 +00:00
|
|
|
public function getHostIPMappings($network)
|
|
|
|
{
|
|
|
|
$addHosts = null;
|
|
|
|
$allContainers = instant_remote_process(["docker network inspect {$network} -f '{{json .Containers}}' "], $this);
|
|
|
|
if (!is_null($allContainers)) {
|
|
|
|
$allContainers = format_docker_command_output_to_json($allContainers);
|
|
|
|
$ips = collect([]);
|
|
|
|
if (count($allContainers) > 0) {
|
|
|
|
$allContainers = $allContainers[0];
|
|
|
|
foreach ($allContainers as $container) {
|
|
|
|
$containerName = data_get($container, 'Name');
|
|
|
|
if ($containerName === 'coolify-proxy') {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
$containerIp = data_get($container, 'IPv4Address');
|
|
|
|
if ($containerName && $containerIp) {
|
|
|
|
$containerIp = str($containerIp)->before('/');
|
|
|
|
$ips->put($containerName, $containerIp->value());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$addHosts = $ips->map(function ($ip, $name) {
|
|
|
|
return "--add-host $name:$ip";
|
|
|
|
})->implode(' ');
|
|
|
|
}
|
|
|
|
return $addHosts;
|
|
|
|
}
|
|
|
|
public function checkIfDockerImageExists(string $imageName, ApplicationDeploymentQueue $deployment)
|
|
|
|
{
|
|
|
|
$this->executeRemoteCommand(
|
|
|
|
commands: collect([
|
|
|
|
[
|
|
|
|
"name" => "local_image_found",
|
|
|
|
"command" => "docker images -q {$imageName} 2>/dev/null",
|
|
|
|
"hidden" => true,
|
|
|
|
]
|
|
|
|
]),
|
|
|
|
loggingModel: $deployment
|
|
|
|
);
|
|
|
|
if (str($deployment->getOutput('local_image_found'))->isEmpty()) {
|
|
|
|
$this->executeRemoteCommand(
|
|
|
|
commands: collect([
|
|
|
|
[
|
|
|
|
"command" => "docker pull {$imageName} 2>/dev/null",
|
|
|
|
"ignoreErrors" => true,
|
|
|
|
"hidden" => true
|
|
|
|
],
|
|
|
|
[
|
|
|
|
"name" => "local_image_found",
|
|
|
|
"command" => "docker images -q {$imageName} 2>/dev/null",
|
|
|
|
"hidden" => true,
|
|
|
|
]
|
|
|
|
]),
|
|
|
|
loggingModel: $deployment
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
public function createWorkDirForDeployment(string $workdir, ApplicationDeploymentQueue $deployment)
|
|
|
|
{
|
|
|
|
$this->executeRemoteCommand(
|
|
|
|
commands: collect([
|
|
|
|
[
|
|
|
|
"command" => executeInDocker($deployment->deployment_uuid, "mkdir -p {$workdir}"),
|
|
|
|
"ignoreErrors" => true,
|
|
|
|
"hidden" => true
|
|
|
|
],
|
|
|
|
]),
|
|
|
|
loggingModel: $deployment
|
|
|
|
);
|
|
|
|
}
|
2023-07-14 11:38:24 +00:00
|
|
|
}
|