feat: deployer
Some checks failed
Deploy / deploy (push) Failing after 21s

This commit is contained in:
ext.jeremy.guillot@maxicoffee.domains
2026-03-10 21:48:18 +01:00
parent 3507349167
commit 3941cb4b8f
3 changed files with 59 additions and 83 deletions

View File

@@ -1,4 +1,4 @@
name: Build and Deploy name: Deploy
on: on:
push: push:
@@ -9,63 +9,22 @@ jobs:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- name: Checkout - name: Checkout
uses: actions/checkout@v4 uses: actions/checkout@v3
- name: Install tools - name: Setup SSH
run: | run: |
apt-get update && apt-get install -y docker.io curl jq mkdir -p ~/.ssh
echo "${{ secrets.DEPLOY_SSH_KEY }}" > ~/.ssh/id_ed25519
chmod 600 ~/.ssh/id_ed25519
ssh-keyscan -H "${{ secrets.DEPLOY_HOST }}" >> ~/.ssh/known_hosts
- name: Build production image - name: Install PHP & Deployer
run: | run: |
docker build --target frankenphp_prod -t mangarr:latest . sudo apt-get install -y php-cli
curl -LO https://deployer.org/deployer.phar
chmod +x deployer.phar
- name: Redeploy via Portainer API - name: Deploy
env: env:
PORTAINER_USER: ${{ secrets.PORTAINER_USER }} DEPLOY_HOST: ${{ secrets.DEPLOY_HOST }}
PORTAINER_PASSWORD: ${{ secrets.PORTAINER_PASSWORD }} run: php deployer.phar deploy production -vvv
run: |
JWT=$(curl -s -X POST http://portainer:9000/api/auth \
-H "Content-Type: application/json" \
-d "{\"Username\":\"$PORTAINER_USER\",\"Password\":\"$PORTAINER_PASSWORD\"}" | jq -r '.jwt')
if [ -z "$JWT" ] || [ "$JWT" = "null" ]; then
echo "Erreur: authentification Portainer echouee"
exit 1
fi
STACK_INFO=$(curl -s http://portainer:9000/api/stacks \
-H "Authorization: Bearer $JWT")
STACK_ID=$(echo "$STACK_INFO" | jq '.[] | select(.Name=="mangarr") | .Id')
ENDPOINT_ID=$(echo "$STACK_INFO" | jq '.[] | select(.Name=="mangarr") | .EndpointId')
if [ -z "$STACK_ID" ] || [ "$STACK_ID" = "null" ]; then
echo "Erreur: stack mangarr non trouvee"
exit 1
fi
echo "Stack ID: $STACK_ID, Endpoint ID: $ENDPOINT_ID"
STACK_FILE=$(curl -s "http://portainer:9000/api/stacks/$STACK_ID/file" \
-H "Authorization: Bearer $JWT" | jq -r '.StackFileContent')
STACK_ENV=$(curl -s "http://portainer:9000/api/stacks/$STACK_ID" \
-H "Authorization: Bearer $JWT" | jq '.Env')
HTTP_CODE=$(curl -s -o /tmp/deploy_result.json -w "%{http_code}" -X PUT \
"http://portainer:9000/api/stacks/$STACK_ID?endpointId=$ENDPOINT_ID" \
-H "Authorization: Bearer $JWT" \
-H "Content-Type: application/json" \
-d "{\"stackFileContent\":$(echo "$STACK_FILE" | jq -Rs .),\"env\":$STACK_ENV,\"prune\":true,\"pullImage\":false}")
echo "Portainer redeploy: HTTP $HTTP_CODE"
if [ "$HTTP_CODE" -ge 300 ]; then
cat /tmp/deploy_result.json
exit 1
fi
- name: Run migrations
run: |
echo "Attente du demarrage de Mangarr..."
sleep 15
docker exec mangarr php bin/console doctrine:migrations:migrate --no-interaction || echo "Rien a migrer"
docker exec mangarr php bin/console cache:clear --env=prod || true
echo "Deploy termine avec succes"

View File

@@ -68,6 +68,19 @@ ENTRYPOINT ["docker-entrypoint"]
HEALTHCHECK --start-period=60s CMD curl -f http://localhost:2019/metrics || exit 1 HEALTHCHECK --start-period=60s CMD curl -f http://localhost:2019/metrics || exit 1
CMD [ "frankenphp", "run", "--config", "/etc/caddy/Caddyfile" ] CMD [ "frankenphp", "run", "--config", "/etc/caddy/Caddyfile" ]
# Runtime FrankenPHP image (sans code baked-in)
# Le code vient du bind mount /srv/mangarr/current:/app (géré par Deployer)
# Builder une seule fois : docker build --target frankenphp_runtime -t mangarr:runtime .
FROM frankenphp_base AS frankenphp_runtime
ENV APP_ENV=prod
ENV FRANKENPHP_CONFIG="import worker.Caddyfile"
RUN mv "$PHP_INI_DIR/php.ini-production" "$PHP_INI_DIR/php.ini"
COPY --link frankenphp/conf.d/app.prod.ini $PHP_INI_DIR/conf.d/
COPY --link frankenphp/worker.Caddyfile /etc/caddy/worker.Caddyfile
# Dev FrankenPHP image # Dev FrankenPHP image
FROM frankenphp_base AS frankenphp_dev FROM frankenphp_base AS frankenphp_dev

View File

@@ -2,38 +2,42 @@
namespace Deployer; namespace Deployer;
require 'recipe/symfony.php'; require 'recipe/symfony.php';
// require 'contrib/webpack_encore.php';
require 'contrib/npm.php';
// Config set('repository', 'https://git.homelab.nestor-server.fr/colgora/Mangarr.git');
set('nodejs_version', 'node_22.x'); set('keep_releases', 3);
set('keep_releases', '3'); set('composer_options', '--no-dev --optimize-autoloader --no-interaction --prefer-dist');
set('repository', 'gitea@git.test.nestor-server.fr:Colgora/Mangarr.git');
set('webpack_encore/env', 'production');
set('composer_options', '--verbose --prefer-dist --no-progress --no-interaction --optimize-autoloader');
set('shared_files', ['.env.local','var/log/prod.log']); // Pas de shared_files ni shared_dirs : tout est géré par les volumes Docker
set('shared_dirs', ['config/secrets','public/cbz','public/tmp','public/images']); set('shared_files', []);
// add('writable_dirs', []); set('shared_dirs', []);
set('writable_dirs', []);
desc('Runs webpack encore build'); host('production')
task('webpack_encore:build', function () { ->set('hostname', getenv('DEPLOY_HOST')) // Injecté depuis le secret Gitea
run("cd {{release_path}} && npm run build"); ->set('remote_user', 'deploy') // User avec accès docker group
}); ->set('deploy_path', '/srv/mangarr')
desc('Run messenger consume');
task('messenger:consume', function () {
run("sudo supervisorctl restart messenger-consume:*");
});
host('mangarr.test.nestor-server.fr')
->set('remote_user', 'colgora')
->set('deploy_path', '/var/www/mangarr')
->set('branch', 'main'); ->set('branch', 'main');
// composer install via container éphémère (pas de PHP sur l'hôte requis)
task('deploy:vendors', function () {
run('docker run --rm -v {{release_path}}:/app -w /app composer:2 install {{composer_options}}');
});
// Build assets via container node éphémère
// mangarr_node_modules volume = cache npm entre les déploiements
desc('Build Webpack Encore assets');
task('webpack_encore:build', function () {
run('docker run --rm -v {{release_path}}:/app -v mangarr_node_modules:/app/node_modules -w /app node:22-alpine sh -c "npm install && npm run build"');
});
// Restart Docker containers (entrypoint gère les migrations automatiquement)
desc('Restart Docker containers');
task('docker:restart', function () {
run('docker restart mangarr-worker-commands mangarr-worker-events mangarr-worker-scheduler');
run('docker restart mangarr');
});
// Hooks // Hooks
after('deploy:vendors', 'npm:install'); after('deploy:vendors', 'webpack_encore:build');
after('npm:install', 'webpack_encore:build'); after('deploy:symlink', 'docker:restart');
after('deploy:vendors', 'database:migrate');
after('deploy:symlink', 'messenger:consume');
after('deploy:failed', 'deploy:unlock'); after('deploy:failed', 'deploy:unlock');