203 lines
8.6 KiB
Groovy
203 lines
8.6 KiB
Groovy
pipeline {
|
||
agent any
|
||
|
||
// 🔁 Vérifie toutes les 2 minutes si du nouveau code a été poussé
|
||
triggers {
|
||
pollSCM('H/2 * * * *')
|
||
}
|
||
|
||
// ⚙️ Paramètre manuel (fallback)
|
||
parameters {
|
||
choice(
|
||
name: 'ENV',
|
||
choices: ['dev', 'preprod', 'prod'],
|
||
description: 'Choisir l’environnement de déploiement (automatique si branche correspondante)'
|
||
)
|
||
}
|
||
|
||
environment {
|
||
REGISTRY_URL = "registry.wk-archi-o24a-15m-g3.fr"
|
||
IMAGE_NAME = "the-tip-top-backend"
|
||
}
|
||
|
||
stages {
|
||
|
||
/* ───────────────────────────────
|
||
* 1️⃣ Init — Détection automatique de l’environnement
|
||
* ─────────────────────────────── */
|
||
stage('Init') {
|
||
steps {
|
||
script {
|
||
def currentBranch = sh(script: "git rev-parse --abbrev-ref HEAD", returnStdout: true).trim()
|
||
echo "🧭 Branche détectée : ${currentBranch}"
|
||
|
||
if (["dev", "preprod", "main"].contains(currentBranch)) {
|
||
env.ENV = (currentBranch == "main") ? "prod" : currentBranch
|
||
} else {
|
||
env.ENV = params.ENV ?: "dev"
|
||
}
|
||
|
||
env.TAG = "${env.ENV}-latest"
|
||
env.DEPLOY_PATH = "/srv/devops/the-tip-top/${env.ENV}"
|
||
|
||
echo """
|
||
🌍 Environnement = ${env.ENV}
|
||
🏷️ Tag Docker = ${env.TAG}
|
||
📂 Chemin de déploiement = ${env.DEPLOY_PATH}
|
||
"""
|
||
|
||
// Vérifie le répertoire cible
|
||
sh "ls -l ${env.DEPLOY_PATH} || echo '⚠️ Dossier non accessible depuis Jenkins'"
|
||
}
|
||
}
|
||
}
|
||
|
||
/* ───────────────────────────────
|
||
* 2️⃣ Checkout du code
|
||
* ─────────────────────────────── */
|
||
stage('Checkout') {
|
||
steps {
|
||
echo "📦 Récupération du code source depuis Gitea..."
|
||
checkout scm
|
||
}
|
||
}
|
||
|
||
/* ───────────────────────────────
|
||
* 3️⃣ Tests & Qualité (dans un conteneur Node.js)
|
||
* ─────────────────────────────── */
|
||
stage('Tests & Qualité') {
|
||
agent {
|
||
docker {
|
||
image 'node:18-alpine'
|
||
args '-u root'
|
||
}
|
||
}
|
||
steps {
|
||
echo "🧪 Lancement des tests et analyse de code..."
|
||
sh '''
|
||
npm ci
|
||
npm run lint || echo "⚠️ Erreurs de lint détectées"
|
||
npm test || echo "⚠️ Tests échoués — vérifier les logs"
|
||
'''
|
||
}
|
||
}
|
||
|
||
/* ───────────────────────────────
|
||
* 4️⃣ Build de l’image Docker
|
||
* ─────────────────────────────── */
|
||
stage('Build Docker image') {
|
||
steps {
|
||
echo "🐳 Construction de l’image Docker backend..."
|
||
sh """
|
||
docker build -t ${REGISTRY_URL}/${IMAGE_NAME}:${TAG} .
|
||
docker tag ${REGISTRY_URL}/${IMAGE_NAME}:${TAG} ${REGISTRY_URL}/${IMAGE_NAME}:latest
|
||
"""
|
||
}
|
||
}
|
||
|
||
/* ───────────────────────────────
|
||
* 5️⃣ Push vers le registre privé
|
||
* ─────────────────────────────── */
|
||
stage('Push to Registry') {
|
||
steps {
|
||
echo "📤 Envoi de l’image vers le registre Docker privé..."
|
||
withCredentials([usernamePassword(credentialsId: 'registry-credentials', usernameVariable: 'REG_USER', passwordVariable: 'REG_PASS')]) {
|
||
sh """
|
||
echo "$REG_PASS" | docker login ${REGISTRY_URL} -u "$REG_USER" --password-stdin
|
||
docker push ${REGISTRY_URL}/${IMAGE_NAME}:${TAG}
|
||
docker push ${REGISTRY_URL}/${IMAGE_NAME}:latest
|
||
"""
|
||
}
|
||
}
|
||
}
|
||
|
||
/* ───────────────────────────────
|
||
* 6️⃣ Backup avant déploiement
|
||
* ─────────────────────────────── */
|
||
stage('Backup Before Deploy') {
|
||
steps {
|
||
echo "💾 Exécution du script de sauvegarde avant déploiement..."
|
||
sh '''
|
||
if [ -f /srv/devops/the-tip-top/backup.sh ]; then
|
||
bash /srv/devops/the-tip-top/backup.sh
|
||
else
|
||
echo "⚠️ Aucun script backup.sh trouvé."
|
||
fi
|
||
'''
|
||
}
|
||
}
|
||
|
||
/* ───────────────────────────────
|
||
* 7️⃣ Déploiement
|
||
* ─────────────────────────────── */
|
||
stage('Deploy') {
|
||
steps {
|
||
echo "🚀 Déploiement du backend sur ${env.ENV}..."
|
||
sh """
|
||
echo "📂 DEPLOY_PATH utilisé : ${DEPLOY_PATH}"
|
||
|
||
if [ ! -f "${DEPLOY_PATH}/docker-compose.yml" ]; then
|
||
echo "❌ Fichier docker-compose.yml introuvable dans ${DEPLOY_PATH}"
|
||
ls -l ${DEPLOY_PATH} || echo "⚠️ Impossible de lister le contenu."
|
||
exit 1
|
||
fi
|
||
|
||
cd "${DEPLOY_PATH}"
|
||
|
||
echo "📦 Pull de l'image Docker depuis le registre..."
|
||
docker compose pull backend
|
||
|
||
echo "🔄 Recréation du conteneur backend..."
|
||
docker compose up -d --force-recreate backend
|
||
"""
|
||
}
|
||
}
|
||
|
||
/* ───────────────────────────────
|
||
* 8️⃣ Vérification de santé (via /health)
|
||
* ─────────────────────────────── */
|
||
stage('Health Check') {
|
||
steps {
|
||
echo "🩺 Vérification du backend après déploiement..."
|
||
script {
|
||
def domain = (env.ENV == 'dev') ? "api.dev.dsp5-archi-o24a-15m-g3.fr" :
|
||
(env.ENV == 'preprod') ? "api.preprod.dsp5-archi-o24a-15m-g3.fr" :
|
||
"api.dsp5-archi-o24a-15m-g3.fr"
|
||
|
||
def statusCode = "000"
|
||
for (int i = 1; i <= 10; i++) {
|
||
// 🔍 Essaie d'abord /health
|
||
statusCode = sh(script: "curl -k -s -o /dev/null -w '%{http_code}' https://${domain}/health || echo 000", returnStdout: true).trim()
|
||
|
||
// 🔁 Si /health renvoie 404, teste /
|
||
if (statusCode == '404') {
|
||
statusCode = sh(script: "curl -k -s -o /dev/null -w '%{http_code}' https://${domain}/ || echo 000", returnStdout: true).trim()
|
||
}
|
||
|
||
if (statusCode in ['200', '301', '302']) {
|
||
echo "✅ Backend ${env.ENV} opérationnel (HTTP ${statusCode})"
|
||
return
|
||
}
|
||
|
||
echo "⏳ Tentative ${i}/10 → HTTP ${statusCode}"
|
||
sleep 5
|
||
}
|
||
error("❌ Health check échoué - code HTTP ${statusCode}")
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
/* ───────────────────────────────
|
||
* 🔚 Post Actions
|
||
* ─────────────────────────────── */
|
||
post {
|
||
success {
|
||
echo "✅ Pipeline backend ${env.ENV} terminé avec succès !"
|
||
}
|
||
failure {
|
||
echo "❌ Échec du pipeline backend pour ${env.ENV}."
|
||
}
|
||
}
|
||
}
|