This commit is contained in:
114
Jenkinsfile
vendored
114
Jenkinsfile
vendored
@@ -7,13 +7,11 @@ pipeline {
|
|||||||
IMAGE_TAG = "${env.BUILD_NUMBER}"
|
IMAGE_TAG = "${env.BUILD_NUMBER}"
|
||||||
K8S_CRED_ID = 'k8s-config'
|
K8S_CRED_ID = 'k8s-config'
|
||||||
|
|
||||||
// Full image refs
|
|
||||||
FRONTEND_IMAGE = '192.168.108.200:80/library/scrum-frontend'
|
FRONTEND_IMAGE = '192.168.108.200:80/library/scrum-frontend'
|
||||||
BACKEND_IMAGE = '192.168.108.200:80/library/scrum-backend'
|
BACKEND_IMAGE = '192.168.108.200:80/library/scrum-backend'
|
||||||
|
|
||||||
// Path to the app inside the workspace
|
// Workspace root IS the project root — no subdirectory needed
|
||||||
APP_DIR = 'scrum-manager'
|
K8S_OVERLAY = 'k8s/overlays/on-premise'
|
||||||
K8S_OVERLAY = 'scrum-manager/k8s/overlays/on-premise'
|
|
||||||
}
|
}
|
||||||
|
|
||||||
options {
|
options {
|
||||||
@@ -24,49 +22,37 @@ pipeline {
|
|||||||
|
|
||||||
stages {
|
stages {
|
||||||
|
|
||||||
// 1. CHECKOUT
|
|
||||||
|
|
||||||
stage('Checkout') {
|
stage('Checkout') {
|
||||||
steps {
|
steps {
|
||||||
checkout scm
|
checkout scm
|
||||||
echo "Building commit: ${env.GIT_COMMIT?.take(8) ?: 'local'}"
|
echo "Workspace: ${env.WORKSPACE}"
|
||||||
|
sh 'ls -la' // quick sanity check — confirm Dockerfile is here
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 2. UNIT TESTS (backend + frontend in parallel)
|
|
||||||
|
|
||||||
stage('Test') {
|
stage('Test') {
|
||||||
parallel {
|
parallel {
|
||||||
stage('Backend Tests') {
|
stage('Backend Tests') {
|
||||||
steps {
|
steps {
|
||||||
dir("${APP_DIR}/server") {
|
dir('server') { // server/ relative to workspace root
|
||||||
sh '''
|
sh 'npm ci && npm test -- --reporter=verbose 2>&1 || true'
|
||||||
npm ci
|
|
||||||
npm test -- --reporter=verbose 2>&1 || true
|
|
||||||
'''
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
stage('Frontend Tests') {
|
stage('Frontend Tests') {
|
||||||
steps {
|
steps {
|
||||||
dir("${APP_DIR}") {
|
// frontend lives at workspace root
|
||||||
sh '''
|
sh 'npm ci && npm test -- --reporter=verbose 2>&1 || true'
|
||||||
npm ci
|
|
||||||
npm test -- --reporter=verbose 2>&1 || true
|
|
||||||
'''
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// 3. BUILD DOCKER IMAGES (parallel)
|
|
||||||
|
|
||||||
stage('Build Images') {
|
stage('Build Images') {
|
||||||
parallel {
|
parallel {
|
||||||
stage('Build Frontend') {
|
stage('Build Frontend') {
|
||||||
steps {
|
steps {
|
||||||
dir("${APP_DIR}") {
|
// Dockerfile is at workspace root
|
||||||
sh """
|
sh """
|
||||||
docker build \
|
docker build \
|
||||||
-f Dockerfile \
|
-f Dockerfile \
|
||||||
@@ -76,10 +62,9 @@ pipeline {
|
|||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
stage('Build Backend') {
|
stage('Build Backend') {
|
||||||
steps {
|
steps {
|
||||||
dir("${APP_DIR}/server") {
|
dir('server') { // server/Dockerfile
|
||||||
sh """
|
sh """
|
||||||
docker build \
|
docker build \
|
||||||
-f Dockerfile \
|
-f Dockerfile \
|
||||||
@@ -93,8 +78,6 @@ pipeline {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 4. PUSH TO HARBOR
|
|
||||||
|
|
||||||
stage('Push to Harbor') {
|
stage('Push to Harbor') {
|
||||||
steps {
|
steps {
|
||||||
withCredentials([usernamePassword(
|
withCredentials([usernamePassword(
|
||||||
@@ -105,11 +88,9 @@ pipeline {
|
|||||||
sh """
|
sh """
|
||||||
echo \$HARBOR_PASS | docker login ${HARBOR_URL} -u \$HARBOR_USER --password-stdin
|
echo \$HARBOR_PASS | docker login ${HARBOR_URL} -u \$HARBOR_USER --password-stdin
|
||||||
|
|
||||||
# Frontend
|
|
||||||
docker push ${FRONTEND_IMAGE}:${IMAGE_TAG}
|
docker push ${FRONTEND_IMAGE}:${IMAGE_TAG}
|
||||||
docker push ${FRONTEND_IMAGE}:latest
|
docker push ${FRONTEND_IMAGE}:latest
|
||||||
|
|
||||||
# Backend
|
|
||||||
docker push ${BACKEND_IMAGE}:${IMAGE_TAG}
|
docker push ${BACKEND_IMAGE}:${IMAGE_TAG}
|
||||||
docker push ${BACKEND_IMAGE}:latest
|
docker push ${BACKEND_IMAGE}:latest
|
||||||
"""
|
"""
|
||||||
@@ -117,15 +98,10 @@ pipeline {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 5. PATCH KUSTOMIZE IMAGE TAGS
|
|
||||||
// (writes the exact build tag into the overlay
|
|
||||||
// so K8s pulls the right image, not :latest)
|
|
||||||
|
|
||||||
stage('Patch Image Tags') {
|
stage('Patch Image Tags') {
|
||||||
steps {
|
steps {
|
||||||
dir("${K8S_OVERLAY}") {
|
dir("${K8S_OVERLAY}") {
|
||||||
sh """
|
sh """
|
||||||
# Requires kustomize v4+ on the Jenkins agent
|
|
||||||
kustomize edit set image \
|
kustomize edit set image \
|
||||||
scrum-frontend=${FRONTEND_IMAGE}:${IMAGE_TAG} \
|
scrum-frontend=${FRONTEND_IMAGE}:${IMAGE_TAG} \
|
||||||
scrum-backend=${BACKEND_IMAGE}:${IMAGE_TAG}
|
scrum-backend=${BACKEND_IMAGE}:${IMAGE_TAG}
|
||||||
@@ -134,66 +110,38 @@ pipeline {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 6. DEPLOY TO KUBERNETES
|
|
||||||
|
|
||||||
stage('Deploy to K8s') {
|
stage('Deploy to K8s') {
|
||||||
steps {
|
steps {
|
||||||
withKubeConfig([credentialsId: "${K8S_CRED_ID}"]) {
|
withKubeConfig([credentialsId: "${K8S_CRED_ID}"]) {
|
||||||
script {
|
|
||||||
echo "Applying kustomize overlay → scrum-manager namespace"
|
|
||||||
|
|
||||||
// Apply the full kustomize overlay (namespace, PV, secrets, deployments, ingress)
|
|
||||||
sh "kubectl apply -k ${K8S_OVERLAY}"
|
sh "kubectl apply -k ${K8S_OVERLAY}"
|
||||||
|
|
||||||
// ── Wait for rollouts ──────────────────────────────
|
sh "kubectl rollout status deployment/mysql -n scrum-manager --timeout=120s"
|
||||||
echo "Waiting for MySQL..."
|
sh "kubectl rollout status deployment/backend -n scrum-manager --timeout=120s"
|
||||||
sh """
|
sh "kubectl rollout status deployment/frontend -n scrum-manager --timeout=120s"
|
||||||
kubectl rollout status deployment/mysql \
|
|
||||||
-n scrum-manager --timeout=120s
|
|
||||||
"""
|
|
||||||
|
|
||||||
echo "Waiting for Backend..."
|
echo "✅ All deployments rolled out."
|
||||||
sh """
|
|
||||||
kubectl rollout status deployment/backend \
|
|
||||||
-n scrum-manager --timeout=120s
|
|
||||||
"""
|
|
||||||
|
|
||||||
echo "Waiting for Frontend..."
|
|
||||||
sh """
|
|
||||||
kubectl rollout status deployment/frontend \
|
|
||||||
-n scrum-manager --timeout=120s
|
|
||||||
"""
|
|
||||||
|
|
||||||
echo "✅ All deployments rolled out successfully."
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// 7. SMOKE TEST (quick sanity check via K8s)
|
|
||||||
|
|
||||||
stage('Smoke Test') {
|
stage('Smoke Test') {
|
||||||
steps {
|
steps {
|
||||||
withKubeConfig([credentialsId: "${K8S_CRED_ID}"]) {
|
withKubeConfig([credentialsId: "${K8S_CRED_ID}"]) {
|
||||||
sh """
|
sh """
|
||||||
# Hit the backend health endpoint from inside the cluster
|
kubectl run smoke-${BUILD_NUMBER} \
|
||||||
# using a one-shot pod so no NodePort/Ingress dependency
|
|
||||||
kubectl run smoke-test-${BUILD_NUMBER} \
|
|
||||||
--image=curlimages/curl:latest \
|
--image=curlimages/curl:latest \
|
||||||
--restart=Never \
|
--restart=Never \
|
||||||
--rm \
|
--rm \
|
||||||
--attach \
|
--attach \
|
||||||
-n scrum-manager \
|
-n scrum-manager \
|
||||||
-- curl -sf http://backend:3001/api/health \
|
-- curl -sf http://backend:3001/api/health \
|
||||||
&& echo "Backend health check PASSED" \
|
&& echo "Health check PASSED" \
|
||||||
|| echo "Backend health check FAILED (non-blocking)"
|
|| echo "Health check FAILED (non-blocking)"
|
||||||
"""
|
"""
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 8. CLEAN UP LOCAL IMAGES
|
|
||||||
|
|
||||||
stage('Clean Up') {
|
stage('Clean Up') {
|
||||||
steps {
|
steps {
|
||||||
sh """
|
sh """
|
||||||
@@ -206,26 +154,26 @@ pipeline {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// POST ACTIONS
|
|
||||||
|
|
||||||
post {
|
post {
|
||||||
success {
|
success {
|
||||||
echo """
|
echo "✅ Build #${env.BUILD_NUMBER} deployed → http://scrum.local"
|
||||||
╔══════════════════════════════════════════════╗
|
|
||||||
║ ✅ scrum-manager deployed successfully ║
|
|
||||||
║ Build : #${env.BUILD_NUMBER} ║
|
|
||||||
║ Tag : ${IMAGE_TAG} ║
|
|
||||||
║ URL : http://scrum.local ║
|
|
||||||
╚══════════════════════════════════════════════╝
|
|
||||||
"""
|
|
||||||
}
|
}
|
||||||
failure {
|
failure {
|
||||||
echo "❌ Pipeline failed. Check logs above."
|
echo "❌ Pipeline failed. Check stage logs above."
|
||||||
// Optional: add mail/Slack notification here
|
|
||||||
}
|
}
|
||||||
always {
|
always {
|
||||||
// Make sure we're logged out of Harbor
|
sh "docker logout ${HARBOR_URL} || true"
|
||||||
sh 'docker logout ${HARBOR_URL} || true'
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**What changed and why:**
|
||||||
|
|
||||||
|
The core fix is that `APP_DIR = 'scrum-manager'` was removed entirely. Your Jenkins job clones the repo and the workspace root lands directly inside the project, giving you this layout:
|
||||||
|
```
|
||||||
|
$WORKSPACE/
|
||||||
|
├── Dockerfile ← frontend image
|
||||||
|
├── server/
|
||||||
|
│ └── Dockerfile ← backend image
|
||||||
|
└── k8s/overlays/...
|
||||||
Reference in New Issue
Block a user