Merge pull request #2568 from matpratta/feat/convert-http-to-ssh-sources-with-deploy-keys

fix: convert HTTP to SSH source when using deploy key on GitHub
This commit is contained in:
Andras Bacsai
2024-11-12 11:32:58 +01:00
committed by GitHub
5 changed files with 273 additions and 174 deletions

View File

@@ -906,21 +906,7 @@ class Application extends BaseModel
public function customRepository()
{
preg_match('/(?<=:)\d+(?=\/)/', $this->git_repository, $matches);
$port = 22;
if (count($matches) === 1) {
$port = $matches[0];
$gitHost = str($this->git_repository)->before(':');
$gitRepo = str($this->git_repository)->after('/');
$repository = "$gitHost:$gitRepo";
} else {
$repository = $this->git_repository;
}
return [
'repository' => $repository,
'port' => $port,
];
return convertGitUrl($this->git_repository, $this->deploymentType(), $this->source);
}
public function generateBaseDir(string $uuid)

View File

@@ -7,6 +7,7 @@ use App\Models\Application;
use App\Models\ApplicationDeploymentQueue;
use App\Models\ApplicationPreview;
use App\Models\EnvironmentVariable;
use App\Models\GithubApp;
use App\Models\InstanceSettings;
use App\Models\LocalFileVolume;
use App\Models\LocalPersistentVolume;
@@ -4092,3 +4093,53 @@ function defaultNginxConfiguration(): string
}
}';
}
function convertGitUrl(string $gitRepository, string $deploymentType, ?GithubApp $source = null): array
{
$repository = $gitRepository;
$providerInfo = [
'host' => null,
'user' => 'git',
'port' => 22,
'repository' => $gitRepository,
];
$sshMatches = [];
$matches = [];
// Let's try and parse the string to detect if it's a valid SSH string or not
preg_match('/((.*?)\:\/\/)?(.*@.*:.*)/', $gitRepository, $sshMatches);
if ($deploymentType === 'deploy_key' && empty($sshMatches) && $source) {
// If this happens, the user may have provided an HTTP URL when they needed an SSH one
// Let's try and fix that for known Git providers
switch ($source->getMorphClass()) {
case \App\Models\GithubApp::class:
$providerInfo['host'] = Url::fromString($source->html_url)->getHost();
$providerInfo['port'] = $source->custom_port;
$providerInfo['user'] = $source->custom_user;
break;
}
if (! empty($providerInfo['host'])) {
// Until we do not support more providers with App (like GithubApp), this will be always true, port will be 22
if ($providerInfo['port'] === 22) {
$repository = "{$providerInfo['user']}@{$providerInfo['host']}:{$providerInfo['repository']}";
} else {
$repository = "ssh://{$providerInfo['user']}@{$providerInfo['host']}:{$providerInfo['port']}/{$providerInfo['repository']}";
}
}
}
preg_match('/(?<=:)\d+(?=\/)/', $gitRepository, $matches);
if (count($matches) === 1) {
$providerInfo['port'] = $matches[0];
$gitHost = str($gitRepository)->before(':');
$gitRepo = str($gitRepository)->after('/');
$repository = "$gitHost:$gitRepo";
}
return [
'repository' => $repository,
'port' => $providerInfo['port'],
];
}

View File

@@ -24,7 +24,7 @@ function logs {
docker exec -t coolify tail -f storage/logs/laravel.log
}
function test {
docker exec -t coolify php artisan test --testsuite=Feature
docker exec -t coolify php artisan test --testsuite=Feature -p
}
function sync:bunny {

View File

@@ -0,0 +1,62 @@
<?php
use App\Models\GithubApp;
test('convertGitUrlsForDeployKeyAndGithubAppAndHttpUrl', function () {
$githubApp = GithubApp::find(0);
$result = convertGitUrl('andrasbacsai/coolify-examples.git', 'deploy_key', $githubApp);
expect($result)->toBe([
'repository' => 'git@github.com:andrasbacsai/coolify-examples.git',
'port' => 22,
]);
});
test('convertGitUrlsForDeployKeyAndGithubAppAndSshUrl', function () {
$githubApp = GithubApp::find(0);
$result = convertGitUrl('git@github.com:andrasbacsai/coolify-examples.git', 'deploy_key', $githubApp);
expect($result)->toBe([
'repository' => 'git@github.com:andrasbacsai/coolify-examples.git',
'port' => 22,
]);
});
test('convertGitUrlsForDeployKeyAndHttpUrl', function () {
$result = convertGitUrl('andrasbacsai/coolify-examples.git', 'deploy_key', null);
expect($result)->toBe([
'repository' => 'andrasbacsai/coolify-examples.git',
'port' => 22,
]);
});
test('convertGitUrlsForDeployKeyAndSshUrl', function () {
$result = convertGitUrl('git@github.com:andrasbacsai/coolify-examples.git', 'deploy_key', null);
expect($result)->toBe([
'repository' => 'git@github.com:andrasbacsai/coolify-examples.git',
'port' => 22,
]);
});
test('convertGitUrlsForSourceAndSshUrl', function () {
$result = convertGitUrl('git@github.com:andrasbacsai/coolify-examples.git', 'source', null);
expect($result)->toBe([
'repository' => 'git@github.com:andrasbacsai/coolify-examples.git',
'port' => 22,
]);
});
test('convertGitUrlsForSourceAndHttpUrl', function () {
$result = convertGitUrl('andrasbacsai/coolify-examples.git', 'source', null);
expect($result)->toBe([
'repository' => 'andrasbacsai/coolify-examples.git',
'port' => 22,
]);
});
test('convertGitUrlsForSourceAndSshUrlWithCustomPort', function () {
$result = convertGitUrl('git@git.domain.com:766/group/project.git', 'source', null);
expect($result)->toBe([
'repository' => 'git@git.domain.com:group/project.git',
'port' => '766',
]);
});

View File

@@ -9,171 +9,171 @@ use App\Models\StandaloneDocker;
use Illuminate\Support\Collection;
use Symfony\Component\Yaml\Yaml;
beforeEach(function () {
$this->applicationYaml = '
version: "3.8"
services:
app:
image: nginx
environment:
SERVICE_FQDN_APP: /app
APP_KEY: base64
APP_DEBUG: "${APP_DEBUG:-false}"
APP_URL: $SERVICE_FQDN_APP
DB_URL: postgres://${SERVICE_USER_POSTGRES}:${SERVICE_PASSWORD_POSTGRES}@db:5432/postgres?schema=public
volumes:
- "./nginx:/etc/nginx"
- "data:/var/www/html"
depends_on:
- db
db:
image: postgres
environment:
POSTGRES_USER: "${SERVICE_USER_POSTGRES}"
POSTGRES_PASSWORD: "${SERVICE_PASSWORD_POSTGRES}"
volumes:
- "dbdata:/var/lib/postgresql/data"
healthcheck:
test:
- CMD
- pg_isready
- "-U"
- "postgres"
interval: 2s
timeout: 10s
retries: 10
depends_on:
app:
condition: service_healthy
networks:
default:
name: something
external: true
noinet:
driver: bridge
internal: true';
// beforeEach(function () {
// $this->applicationYaml = '
// version: "3.8"
// services:
// app:
// image: nginx
// environment:
// SERVICE_FQDN_APP: /app
// APP_KEY: base64
// APP_DEBUG: "${APP_DEBUG:-false}"
// APP_URL: $SERVICE_FQDN_APP
// DB_URL: postgres://${SERVICE_USER_POSTGRES}:${SERVICE_PASSWORD_POSTGRES}@db:5432/postgres?schema=public
// volumes:
// - "./nginx:/etc/nginx"
// - "data:/var/www/html"
// depends_on:
// - db
// db:
// image: postgres
// environment:
// POSTGRES_USER: "${SERVICE_USER_POSTGRES}"
// POSTGRES_PASSWORD: "${SERVICE_PASSWORD_POSTGRES}"
// volumes:
// - "dbdata:/var/lib/postgresql/data"
// healthcheck:
// test:
// - CMD
// - pg_isready
// - "-U"
// - "postgres"
// interval: 2s
// timeout: 10s
// retries: 10
// depends_on:
// app:
// condition: service_healthy
// networks:
// default:
// name: something
// external: true
// noinet:
// driver: bridge
// internal: true';
$this->applicationComposeFileString = Yaml::parse($this->applicationYaml);
// $this->applicationComposeFileString = Yaml::parse($this->applicationYaml);
$this->application = Application::create([
'name' => 'Application for tests',
'docker_compose_domains' => json_encode([
'app' => [
'domain' => 'http://bcoowoookw0co4cok4sgc4k8.127.0.0.1.sslip.io',
],
]),
'preview_url_template' => '{{pr_id}}.{{domain}}',
'uuid' => 'bcoowoookw0co4cok4sgc4k8s',
'repository_project_id' => 603035348,
'git_repository' => 'coollabsio/coolify-examples',
'git_branch' => 'main',
'base_directory' => '/docker-compose-test',
'docker_compose_location' => 'docker-compose.yml',
'docker_compose_raw' => $this->applicationYaml,
'build_pack' => 'dockercompose',
'ports_exposes' => '3000',
'environment_id' => 1,
'destination_id' => 0,
'destination_type' => StandaloneDocker::class,
'source_id' => 1,
'source_type' => GithubApp::class,
]);
$this->application->environment_variables_preview()->where('key', 'APP_DEBUG')->update(['value' => 'true']);
$this->applicationPreview = ApplicationPreview::create([
'git_type' => 'github',
'application_id' => $this->application->id,
'pull_request_id' => 1,
'pull_request_html_url' => 'https://github.com/coollabsio/coolify-examples/pull/1',
]);
$this->serviceYaml = '
services:
activepieces:
image: "ghcr.io/activepieces/activepieces:latest"
environment:
- SERVICE_FQDN_ACTIVEPIECES
- AP_API_KEY=$SERVICE_PASSWORD_64_APIKEY
- AP_URL=$SERVICE_URL_ACTIVEPIECES
- AP_ENCRYPTION_KEY=$SERVICE_PASSWORD_ENCRYPTIONKEY
- AP_ENGINE_EXECUTABLE_PATH=dist/packages/engine/main.js
- AP_ENVIRONMENT=prod
- AP_EXECUTION_MODE=${AP_EXECUTION_MODE}
- AP_FRONTEND_URL=$SERVICE_FQDN_ACTIVEPIECES
- AP_JWT_SECRET=$SERVICE_PASSWORD_64_JWT
- AP_POSTGRES_DATABASE=activepieces
- AP_POSTGRES_HOST=postgres
- AP_POSTGRES_PASSWORD=$SERVICE_PASSWORD_POSTGRES
- AP_POSTGRES_PORT=5432
- AP_POSTGRES_USERNAME=$SERVICE_USER_POSTGRES
- AP_REDIS_HOST=redis
- AP_REDIS_PORT=6379
- AP_SANDBOX_RUN_TIME_SECONDS=600
- AP_TELEMETRY_ENABLED=true
- "AP_TEMPLATES_SOURCE_URL=https://cloud.activepieces.com/api/v1/flow-templates"
- AP_TRIGGER_DEFAULT_POLL_INTERVAL=5
- AP_WEBHOOK_TIMEOUT_SECONDS=30
depends_on:
postgres:
condition: service_healthy
redis:
condition: service_started
healthcheck:
test: ["CMD", "curl", "-f", "http://127.0.0.1:80"]
interval: 5s
timeout: 20s
retries: 10
postgres:
image: "nginx"
environment:
- SERVICE_FQDN_ACTIVEPIECES=/api
- POSTGRES_DB=activepieces
- PASSW=$AP_POSTGRES_PASSWORD
- AP_FRONTEND_URL=$SERVICE_FQDN_ACTIVEPIECES
- POSTGRES_PASSWORD=$SERVICE_PASSWORD_POSTGRES
- POSTGRES_USER=$SERVICE_USER_POSTGRES
volumes:
- "pg-data:/var/lib/postgresql/data"
healthcheck:
test: ["CMD-SHELL", "pg_isready -U $${POSTGRES_USER} -d $${POSTGRES_DB}"]
interval: 5s
timeout: 20s
retries: 10
redis:
image: "redis:latest"
volumes:
- "redis_data:/data"
healthcheck:
test: ["CMD", "redis-cli", "ping"]
interval: 5s
timeout: 20s
retries: 10
// $this->application = Application::create([
// 'name' => 'Application for tests',
// 'docker_compose_domains' => json_encode([
// 'app' => [
// 'domain' => 'http://bcoowoookw0co4cok4sgc4k8.127.0.0.1.sslip.io',
// ],
// ]),
// 'preview_url_template' => '{{pr_id}}.{{domain}}',
// 'uuid' => 'bcoowoookw0co4cok4sgc4k8s',
// 'repository_project_id' => 603035348,
// 'git_repository' => 'coollabsio/coolify-examples',
// 'git_branch' => 'main',
// 'base_directory' => '/docker-compose-test',
// 'docker_compose_location' => 'docker-compose.yml',
// 'docker_compose_raw' => $this->applicationYaml,
// 'build_pack' => 'dockercompose',
// 'ports_exposes' => '3000',
// 'environment_id' => 1,
// 'destination_id' => 0,
// 'destination_type' => StandaloneDocker::class,
// 'source_id' => 1,
// 'source_type' => GithubApp::class,
// ]);
// $this->application->environment_variables_preview()->where('key', 'APP_DEBUG')->update(['value' => 'true']);
// $this->applicationPreview = ApplicationPreview::create([
// 'git_type' => 'github',
// 'application_id' => $this->application->id,
// 'pull_request_id' => 1,
// 'pull_request_html_url' => 'https://github.com/coollabsio/coolify-examples/pull/1',
// ]);
// $this->serviceYaml = '
// services:
// activepieces:
// image: "ghcr.io/activepieces/activepieces:latest"
// environment:
// - SERVICE_FQDN_ACTIVEPIECES
// - AP_API_KEY=$SERVICE_PASSWORD_64_APIKEY
// - AP_URL=$SERVICE_URL_ACTIVEPIECES
// - AP_ENCRYPTION_KEY=$SERVICE_PASSWORD_ENCRYPTIONKEY
// - AP_ENGINE_EXECUTABLE_PATH=dist/packages/engine/main.js
// - AP_ENVIRONMENT=prod
// - AP_EXECUTION_MODE=${AP_EXECUTION_MODE}
// - AP_FRONTEND_URL=$SERVICE_FQDN_ACTIVEPIECES
// - AP_JWT_SECRET=$SERVICE_PASSWORD_64_JWT
// - AP_POSTGRES_DATABASE=activepieces
// - AP_POSTGRES_HOST=postgres
// - AP_POSTGRES_PASSWORD=$SERVICE_PASSWORD_POSTGRES
// - AP_POSTGRES_PORT=5432
// - AP_POSTGRES_USERNAME=$SERVICE_USER_POSTGRES
// - AP_REDIS_HOST=redis
// - AP_REDIS_PORT=6379
// - AP_SANDBOX_RUN_TIME_SECONDS=600
// - AP_TELEMETRY_ENABLED=true
// - "AP_TEMPLATES_SOURCE_URL=https://cloud.activepieces.com/api/v1/flow-templates"
// - AP_TRIGGER_DEFAULT_POLL_INTERVAL=5
// - AP_WEBHOOK_TIMEOUT_SECONDS=30
// depends_on:
// postgres:
// condition: service_healthy
// redis:
// condition: service_started
// healthcheck:
// test: ["CMD", "curl", "-f", "http://127.0.0.1:80"]
// interval: 5s
// timeout: 20s
// retries: 10
// postgres:
// image: "nginx"
// environment:
// - SERVICE_FQDN_ACTIVEPIECES=/api
// - POSTGRES_DB=activepieces
// - PASSW=$AP_POSTGRES_PASSWORD
// - AP_FRONTEND_URL=$SERVICE_FQDN_ACTIVEPIECES
// - POSTGRES_PASSWORD=$SERVICE_PASSWORD_POSTGRES
// - POSTGRES_USER=$SERVICE_USER_POSTGRES
// volumes:
// - "pg-data:/var/lib/postgresql/data"
// healthcheck:
// test: ["CMD-SHELL", "pg_isready -U $${POSTGRES_USER} -d $${POSTGRES_DB}"]
// interval: 5s
// timeout: 20s
// retries: 10
// redis:
// image: "redis:latest"
// volumes:
// - "redis_data:/data"
// healthcheck:
// test: ["CMD", "redis-cli", "ping"]
// interval: 5s
// timeout: 20s
// retries: 10
';
// ';
$this->serviceComposeFileString = Yaml::parse($this->serviceYaml);
// $this->serviceComposeFileString = Yaml::parse($this->serviceYaml);
$this->service = Service::create([
'name' => 'Service for tests',
'uuid' => 'tgwcg8w4s844wkog8kskw44g',
'docker_compose_raw' => $this->serviceYaml,
'environment_id' => 1,
'server_id' => 0,
'destination_id' => 0,
'destination_type' => StandaloneDocker::class,
]);
});
// $this->service = Service::create([
// 'name' => 'Service for tests',
// 'uuid' => 'tgwcg8w4s844wkog8kskw44g',
// 'docker_compose_raw' => $this->serviceYaml,
// 'environment_id' => 1,
// 'server_id' => 0,
// 'destination_id' => 0,
// 'destination_type' => StandaloneDocker::class,
// ]);
// });
afterEach(function () {
// $this->applicationPreview->forceDelete();
$this->application->forceDelete();
DeleteResourceJob::dispatchSync($this->service);
$this->service->forceDelete();
});
// afterEach(function () {
// // $this->applicationPreview->forceDelete();
// $this->application->forceDelete();
// DeleteResourceJob::dispatchSync($this->service);
// $this->service->forceDelete();
// });
test('ServiceComposeParseNew', function () {
$output = newParser($this->service);
$this->service->saveComposeConfigs();
expect($output)->toBeInstanceOf(Collection::class);
});
// test('ServiceComposeParseNew', function () {
// $output = newParser($this->service);
// $this->service->saveComposeConfigs();
// expect($output)->toBeInstanceOf(Collection::class);
// });
// test('ApplicationComposeParse', function () {
// expect($this->jsonapplicationComposeFile)->toBeJson()->ray();