// deployment/Jenkinsfile_ArgoCD def PIPELINE_ID = "${env.BUILD_NUMBER}" def getImageTag() { def dateFormat = new java.text.SimpleDateFormat('yyyyMMddHHmmss') def currentDate = new Date() return dateFormat.format(currentDate) } podTemplate( label: "${PIPELINE_ID}", serviceAccount: 'jenkins', containers: [ containerTemplate(name: 'node', image: 'node:20-slim', ttyEnabled: true, command: 'cat'), containerTemplate(name: 'podman', image: "mgoltzsche/podman", ttyEnabled: true, command: 'cat', privileged: true), containerTemplate(name: 'git', image: 'alpine/git:latest', command: 'cat', ttyEnabled: true) ], volumes: [ emptyDirVolume(mountPath: '/run/podman', memory: false) ] ) { node(PIPELINE_ID) { def props def imageTag = getImageTag() // Manifest Repository 설정 def MANIFEST_REPO = 'https://github.com/won-ktds/smarketing-manifest.git' def MANIFEST_CREDENTIAL_ID = 'github-credentials-smarketing' try { stage("Get Source") { checkout scm // 환경변수 파일 확인 및 읽기 if (!fileExists('deployment/deploy_env_vars')) { error "deployment/deploy_env_vars 파일이 없습니다!" } props = readProperties file: "deployment/deploy_env_vars" // 필수 환경변수 검증 if (!props.registry || !props.image_org || !props.namespace) { error "필수 환경변수가 누락되었습니다. registry, image_org, namespace를 확인하세요." } echo "=== Build Information ===" echo "Service: smarketing-frontend" echo "Image Tag: ${imageTag}" echo "Registry: ${props.registry}" echo "Image Org: ${props.image_org}" echo "Namespace: ${props.namespace}" } stage("Check Changes") { script { def changes = sh( script: "git diff --name-only HEAD~1 HEAD", returnStdout: true ).trim() if (!changes.contains("src/") && !changes.contains("public/") && !changes.contains("package")) { echo "No significant frontend changes detected, skipping build" currentBuild.result = 'SUCCESS' error("Stopping pipeline - no frontend changes detected") } echo "Frontend changes detected, proceeding with build" } } stage('Build & Push Frontend Image') { container('podman') { sh 'podman system service -t 0 unix:///run/podman/podman.sock & sleep 2' withCredentials([usernamePassword( credentialsId: 'acr-credentials', usernameVariable: 'ACR_USERNAME', passwordVariable: 'ACR_PASSWORD' )]) { def imagePath = "${props.registry}/${props.image_org}/smarketing-frontend:${imageTag}" sh """ echo "==========================================" echo "Building smarketing-frontend" echo "Image Tag: ${imageTag}" echo "Image Path: ${imagePath}" echo "==========================================" # ACR 로그인 echo \$ACR_PASSWORD | podman login ${props.registry} --username \$ACR_USERNAME --password-stdin # Docker 이미지 빌드 podman build \\ --build-arg PROJECT_FOLDER="." \\ --build-arg VUE_APP_AUTH_URL="${props.auth_url}" \\ --build-arg VUE_APP_MEMBER_URL="${props.member_url}" \\ --build-arg VUE_APP_STORE_URL="${props.store_url}" \\ --build-arg VUE_APP_MENU_URL="${props.menu_url}" \\ --build-arg VUE_APP_SALES_URL="${props.sales_url}" \\ --build-arg VUE_APP_CONTENT_URL="${props.content_url}" \\ --build-arg VUE_APP_RECOMMEND_URL="${props.recommend_url}" \\ --build-arg BUILD_FOLDER="deployment/container" \\ --build-arg EXPORT_PORT="${props.export_port}" \\ -f deployment/container/Dockerfile-smarketing-frontend \\ -t ${imagePath} . # 이미지 푸시 podman push ${imagePath} echo "✅ Frontend image pushed successfully: ${imagePath}" """ } } } stage('Update Manifest Repository') { container('git') { script { // Manifest Repository Clone withCredentials([usernamePassword( credentialsId: MANIFEST_CREDENTIAL_ID, usernameVariable: 'GIT_USERNAME', passwordVariable: 'GIT_PASSWORD' )]) { sh """ echo "=== Git 설정 ===" git config --global user.name "Jenkins CI" git config --global user.email "jenkins@company.com" echo "=== Manifest Repository Clone ===" rm -rf manifest-repo git clone https://\$GIT_USERNAME:\$GIT_PASSWORD@github.com/won-ktds/smarketing-manifest.git manifest-repo cd manifest-repo """ def fullImageName = "${props.registry}/${props.image_org}/smarketing-frontend:${imageTag}" def deploymentFile = "smarketing-frontend/deployments/frontend-deployment.yaml" sh """ cd manifest-repo echo "=== smarketing-frontend 이미지 태그 업데이트 ===" if [ -f "${deploymentFile}" ]; then # 이미지 태그 업데이트 (sed 사용) sed -i 's|image: ${props.registry}/${props.image_org}/smarketing-frontend:.*|image: ${fullImageName}|g' "${deploymentFile}" echo "Updated ${deploymentFile} with new image: ${fullImageName}" # 변경사항 확인 echo "=== 변경된 내용 확인 ===" grep "image: ${props.registry}/${props.image_org}/smarketing-frontend" "${deploymentFile}" || echo "이미지 태그 업데이트 확인 실패" else echo "Warning: ${deploymentFile} not found" echo "Creating manifest directory structure..." # 기본 구조 생성 mkdir -p smarketing-frontend/deployments # 기본 deployment 파일 생성 cat > "${deploymentFile}" << EOF --- apiVersion: v1 kind: ConfigMap metadata: name: smarketing-frontend-config namespace: ${props.namespace} data: runtime-env.js: | window.__runtime_config__ = { AUTH_URL: '${props.auth_url}', MEMBER_URL: '${props.member_url}', STORE_URL: '${props.store_url}', MENU_URL: '${props.menu_url}', SALES_URL: '${props.sales_url}', CONTENT_URL: '${props.content_url}', RECOMMEND_URL: '${props.recommend_url}', GATEWAY_URL: 'http://${props.ingress_host}', ENV: 'production', DEBUG: false }; --- apiVersion: apps/v1 kind: Deployment metadata: name: smarketing-frontend namespace: ${props.namespace} labels: app: smarketing-frontend spec: replicas: ${props.replicas} selector: matchLabels: app: smarketing-frontend template: metadata: labels: app: smarketing-frontend spec: imagePullSecrets: - name: acr-secret containers: - name: smarketing-frontend image: ${fullImageName} imagePullPolicy: Always ports: - containerPort: ${props.export_port} resources: requests: cpu: ${props.resources_requests_cpu} memory: ${props.resources_requests_memory} limits: cpu: ${props.resources_limits_cpu} memory: ${props.resources_limits_memory} volumeMounts: - name: runtime-config mountPath: /usr/share/nginx/html/runtime-env.js subPath: runtime-env.js livenessProbe: httpGet: path: /health port: ${props.export_port} initialDelaySeconds: 30 periodSeconds: 10 readinessProbe: httpGet: path: /health port: ${props.export_port} initialDelaySeconds: 5 periodSeconds: 5 volumes: - name: runtime-config configMap: name: smarketing-frontend-config --- apiVersion: v1 kind: Service metadata: name: smarketing-frontend-service namespace: ${props.namespace} labels: app: smarketing-frontend spec: type: LoadBalancer ports: - port: 80 targetPort: ${props.export_port} protocol: TCP name: http selector: app: smarketing-frontend --- apiVersion: networking.k8s.io/v1 kind: Ingress metadata: name: smarketing-frontend-ingress namespace: ${props.namespace} annotations: nginx.ingress.kubernetes.io/rewrite-target: / nginx.ingress.kubernetes.io/ssl-redirect: "false" spec: rules: - host: ${props.ingress_host} http: paths: - path: / pathType: Prefix backend: service: name: smarketing-frontend-service port: number: 80 EOF echo "Created basic frontend-deployment.yaml" fi """ sh """ cd manifest-repo echo "=== Git 변경사항 확인 ===" git status git diff # 변경사항이 있으면 커밋 및 푸시 if [ -n "\$(git status --porcelain)" ]; then git add . git commit -m "Update smarketing-frontend to ${imageTag} - Build ${env.BUILD_NUMBER}" git push origin main echo "✅ Successfully updated manifest repository" else echo "ℹ️ No changes to commit" fi """ } } } } stage('Trigger ArgoCD Sync') { script { echo """ 🎯 Frontend CI Pipeline 완료! 📦 빌드된 이미지: - ${props.registry}/${props.image_org}/smarketing-frontend:${imageTag} 🔄 ArgoCD 동작: - ArgoCD가 manifest repository 변경사항을 자동으로 감지합니다 - smarketing-frontend Application이 새로운 이미지로 동기화됩니다 - ArgoCD UI에서 배포 상태를 모니터링하세요 🌐 ArgoCD UI: [ArgoCD 접속 URL] 📁 Manifest Repo: ${MANIFEST_REPO} """ } } // 성공 시 처리 echo """ ✅ Frontend CI Pipeline 성공! 🏷️ 새로운 이미지 태그: ${imageTag} 🔄 ArgoCD가 자동으로 배포를 시작합니다 """ } catch (Exception e) { // 실패 시 처리 echo "❌ Frontend CI Pipeline 실패: ${e.getMessage()}" throw e } finally { // 정리 작업 (항상 실행) container('podman') { sh 'podman system prune -f || true' } sh 'rm -rf manifest-repo || true' } } }