| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289 |
- pipeline {
- agent any
- parameters {
- choice(name: 'env', choices: ['dev', 'test', 'prod'], description: '部署环境')
- string(name: 'NAMESPACE', defaultValue: 'portal-frontend', description: 'K8s 命名空间')
- booleanParam(name: 'FORCE_UPDATE_CONFIG', defaultValue: false, description: '强制更新配置(包括 Deployment 和 Ingress)')
- }
- environment {
- PROJECT_NAME = 'portal-frontend'
- BUILD_DIR = 'dist'
- NODE_ENV = 'production'
- HARBOR_HOST = '8.130.28.21:81'
- KUBECONFIG_PATH = '/root/.kube/config'
- HARBOR_USER = 'admin'
- HARBOR_PASS = 'Hfln@1024'
- HARBOR_RETENTION_ID = '1'
- DOMAIN = 'radar-power.asia'
- TLS_SECRET = 'portal-tls'
- }
- stages {
- stage('🧬 初始化环境') {
- steps {
- script {
- env.HARBOR_PROJECT = params.env
- env.IMAGE_TAG = "${HARBOR_HOST}/${env.HARBOR_PROJECT}/${PROJECT_NAME}:${BUILD_NUMBER}"
- echo ">>> 环境:${params.env}, Harbor项目:${env.HARBOR_PROJECT}, K8s命名空间:${params.NAMESPACE}"
- echo ">>> 域名:${env.DOMAIN}, TLS Secret:${env.TLS_SECRET}"
- echo ">>> 强制更新配置:${params.FORCE_UPDATE_CONFIG}"
- }
- }
- }
- stage('📥 拉取代码') {
- steps {
- checkout scm
- echo "✅ 代码拉取成功"
- }
- }
- stage('🔐 配置 Ingress 控制器') {
- steps {
- script {
- sh """
- export KUBECONFIG=${KUBECONFIG_PATH}
- echo ">>> 配置 Ingress 控制器为 LoadBalancer 类型..."
- kubectl patch svc ingress-nginx-controller -n ingress-nginx -p '{"spec":{"type":"LoadBalancer"}}' || echo "⚠️ 修改失败,可能已经是 LoadBalancer 类型"
- echo ">>> 当前 Ingress 控制器状态:"
- kubectl get svc ingress-nginx-controller -n ingress-nginx
- """
- }
- }
- }
- stage('🔧 构建 Docker 镜像') {
- steps {
- script {
- sh """
- docker login -u ${HARBOR_USER} -p ${HARBOR_PASS} ${HARBOR_HOST}
- docker build --build-arg ENV=${params.env} -t ${IMAGE_TAG} .
- """
- echo "✅ 镜像构建成功:${IMAGE_TAG}"
- }
- }
- }
- stage('🚀 推送镜像到 Harbor') {
- steps {
- script {
- sh """
- docker push ${IMAGE_TAG}
- docker rmi ${IMAGE_TAG}
- """
- echo "✅ 镜像推送并本地清理完成"
- }
- }
- }
- stage(' Kubernetes 部署') {
- steps {
- script {
- sh """
- export KUBECONFIG=${KUBECONFIG_PATH}
- # 确保命名空间存在
- kubectl get ns ${params.NAMESPACE} >/dev/null 2>&1 || kubectl create ns ${params.NAMESPACE}
- # 如果强制更新配置或 Deployment 不存在,则重新创建
- if [ "${params.FORCE_UPDATE_CONFIG}" = "true" ] || ! kubectl get deployment ${PROJECT_NAME} -n ${params.NAMESPACE} >/dev/null 2>&1; then
- if [ "${params.FORCE_UPDATE_CONFIG}" = "true" ]; then
- echo ">>> 强制更新配置,删除现有资源..."
- kubectl delete deployment ${PROJECT_NAME} -n ${params.NAMESPACE} --ignore-not-found=true
- kubectl delete svc ${PROJECT_NAME} -n ${params.NAMESPACE} --ignore-not-found=true
- kubectl delete ingress ${PROJECT_NAME}-ingress -n ${params.NAMESPACE} --ignore-not-found=true
- fi
- echo ">>> 创建 Deployment 和 Service..."
- kubectl apply -n ${params.NAMESPACE} -f - <<EOF
- apiVersion: apps/v1
- kind: Deployment
- metadata:
- name: ${PROJECT_NAME}
- spec:
- replicas: 2
- selector:
- matchLabels:
- app: ${PROJECT_NAME}
- template:
- metadata:
- labels:
- app: ${PROJECT_NAME}
- spec:
- containers:
- - name: ${PROJECT_NAME}
- image: ${IMAGE_TAG}
- ports:
- - containerPort: 80
- resources:
- requests:
- memory: "128Mi"
- cpu: "100m"
- limits:
- memory: "256Mi"
- cpu: "200m"
- livenessProbe:
- httpGet:
- path: /
- port: 80
- initialDelaySeconds: 30
- periodSeconds: 10
- readinessProbe:
- httpGet:
- path: /
- port: 80
- initialDelaySeconds: 5
- periodSeconds: 5
- ---
- apiVersion: v1
- kind: Service
- metadata:
- name: ${PROJECT_NAME}
- spec:
- type: ClusterIP
- selector:
- app: ${PROJECT_NAME}
- ports:
- - port: 80
- targetPort: 80
- protocol: TCP
- EOF
- echo ">>> 创建 Ingress..."
- kubectl apply -n ${params.NAMESPACE} -f - <<EOF
- apiVersion: networking.k8s.io/v1
- kind: Ingress
- metadata:
- name: ${PROJECT_NAME}-ingress
- annotations:
- kubernetes.io/ingress.class: nginx
- nginx.ingress.kubernetes.io/ssl-redirect: "true"
- nginx.ingress.kubernetes.io/force-ssl-redirect: "true"
- nginx.ingress.kubernetes.io/backend-protocol: "HTTP"
- nginx.ingress.kubernetes.io/ssl-passthrough: "false"
- nginx.ingress.kubernetes.io/proxy-body-size: "8m"
- nginx.ingress.kubernetes.io/rewrite-target: /
- nginx.ingress.kubernetes.io/ssl-ciphers: "ECDHE-RSA-AES128-GCM-SHA256:ECDHE-RSA-AES256-GCM-SHA384:ECDHE-RSA-AES128-SHA256:ECDHE-RSA-AES256-SHA384"
- nginx.ingress.kubernetes.io/ssl-protocols: "TLSv1.2 TLSv1.3"
- spec:
- tls:
- - hosts:
- - ${env.DOMAIN}
- secretName: ${env.TLS_SECRET}
- rules:
- - host: ${env.DOMAIN}
- http:
- paths:
- - path: /
- pathType: Prefix
- backend:
- service:
- name: ${PROJECT_NAME}
- port:
- number: 80
- EOF
- else
- # 只更新镜像
- echo ">>> Deployment 已存在,仅更新镜像..."
- kubectl set image deployment/${PROJECT_NAME} ${PROJECT_NAME}=${IMAGE_TAG} -n ${params.NAMESPACE}
- echo "✅ 镜像更新完成"
- fi
- # 显示部署状态
- echo ">>> 部署状态:"
- kubectl get all -n ${params.NAMESPACE}
- kubectl get ingress -n ${params.NAMESPACE}
- echo "✅ 应用部署完成!"
- echo "�� 访问地址:https://${env.DOMAIN}"
- echo " 注意:请确保域名 ${env.DOMAIN} 已正确解析到集群"
- """
- }
- }
- }
- stage('🔍 部署验证') {
- steps {
- script {
- sh """
- export KUBECONFIG=${KUBECONFIG_PATH}
- echo ">>> 验证部署状态..."
- # 检查 Pod 状态
- kubectl get pods -n ${params.NAMESPACE} -l app=${PROJECT_NAME}
- # 检查 Service 状态
- kubectl get svc -n ${params.NAMESPACE} ${PROJECT_NAME}
- # 检查 Ingress 状态
- kubectl get ingress -n ${params.NAMESPACE} ${PROJECT_NAME}-ingress
- # 检查 TLS Secret
- kubectl get secret -n ${params.NAMESPACE} ${env.TLS_SECRET}
- echo "✅ 部署验证完成"
- """
- }
- }
- }
- stage('🧹 清理本地旧镜像(保留最新3个)') {
- steps {
- script {
- def baseImage = "${HARBOR_HOST}/${env.HARBOR_PROJECT}/${PROJECT_NAME}"
- sh """
- docker images ${baseImage} --format "{{.Repository}}:{{.Tag}}" \\
- | grep -v latest \\
- | sort -r -t ':' -k2 \\
- | tail -n +4 \\
- | xargs -r docker rmi || true
- """
- echo "✅ 本地旧镜像清理完成"
- }
- }
- }
- stage(' 清理悬空镜像 <none>') {
- steps {
- script {
- sh """
- docker images -f "dangling=true" -q | xargs -r docker rmi || true
- """
- echo "✅ 悬空镜像(<none>)清理完成"
- }
- }
- }
- stage(' 触发 Harbor 镜像保留策略') {
- steps {
- script {
- sh """
- curl -u ${HARBOR_USER}:${HARBOR_PASS} -X POST \\
- "http://${HARBOR_HOST}/api/v2.0/retentions/${HARBOR_RETENTION_ID}/executions"
- """
- echo "✅ Harbor 镜像保留策略已触发"
- }
- }
- }
- }
- post {
- success {
- echo "✅ 构建 & 部署成功!"
- echo "🌐 应用可通过 https://${env.DOMAIN} 访问"
- }
- failure {
- echo "❌ 构建或部署失败,请检查日志"
- }
- always {
- cleanWs()
- }
- }
- }
|