Jenkinsfile 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. pipeline {
  2. agent any
  3. parameters {
  4. choice(name: 'env', choices: ['dev', 'test', 'prod'], description: '部署环境(dev/test/prod)')
  5. string(name: 'NAMESPACE', defaultValue: 'hfln-dev', description: 'Kubernetes 命名空间')
  6. string(name: 'INGRESS_HOST', defaultValue: '', description: 'Ingress 域名(为空则不创建Ingress)')
  7. }
  8. environment {
  9. PROJECT_NAME = 'device-service'
  10. MODULE_NAME = 'device-service-server'
  11. MAVEN_HOME = '/usr/local/apache-maven-3.9.9/bin'
  12. HARBOR_HOST = '8.130.28.21:81'
  13. KUBECONFIG_PATH = '/root/.kube/config'
  14. SPRING_PROFILES_ACTIVE = "${params.env}"
  15. }
  16. stages {
  17. stage('🧬 设置环境变量') {
  18. steps {
  19. script {
  20. if (params.env == 'prod') {
  21. env.HARBOR_PROJECT = 'prod'
  22. } else if (params.env == 'test') {
  23. env.HARBOR_PROJECT = 'test'
  24. } else {
  25. env.HARBOR_PROJECT = 'dev'
  26. }
  27. echo ">>> 使用环境:${params.env},HARBOR 项目:${env.HARBOR_PROJECT},K8S 命名空间:${params.NAMESPACE},Ingress 域名:${params.INGRESS_HOST ?: '无'}"
  28. }
  29. }
  30. }
  31. stage('🧬 拉取 Git 代码') {
  32. steps {
  33. script {
  34. echo ">>> 拉取代码中..."
  35. checkout scm
  36. echo ">>> 代码拉取完成 ✅"
  37. }
  38. }
  39. }
  40. stage('📦 Maven 构建项目') {
  41. steps {
  42. script {
  43. echo ">>> 开始构建模块:${MODULE_NAME}"
  44. sh "${MAVEN_HOME}/mvn clean package -DskipTests -pl ${MODULE_NAME} -am"
  45. echo ">>> 构建完成 ✅"
  46. }
  47. }
  48. }
  49. stage('🐳 构建 & 推送 Docker 镜像') {
  50. steps {
  51. script {
  52. def imageTag = "${HARBOR_HOST}/${env.HARBOR_PROJECT}/${PROJECT_NAME}:${BUILD_NUMBER}"
  53. echo ">>> 构建 Docker 镜像: ${imageTag}"
  54. sh """
  55. docker login -u admin -p Hfln@1024 ${HARBOR_HOST}
  56. docker build --build-arg spring_profiles_active=${params.env} -t ${imageTag} .
  57. docker push ${imageTag}
  58. docker rmi ${imageTag}
  59. """
  60. echo ">>> Docker 镜像构建并推送完成 ✅"
  61. }
  62. }
  63. }
  64. stage('🚀 部署到 Kubernetes') {
  65. steps {
  66. script {
  67. def imageTag = "${HARBOR_HOST}/${env.HARBOR_PROJECT}/${PROJECT_NAME}:${BUILD_NUMBER}"
  68. echo ">>> 正在部署:${PROJECT_NAME} 到命名空间:${params.NAMESPACE}"
  69. sh """
  70. export KUBECONFIG=${KUBECONFIG_PATH}
  71. # 创建命名空间(如果不存在)
  72. kubectl get ns ${params.NAMESPACE} || kubectl create ns ${params.NAMESPACE}
  73. # 部署 Deployment(2 副本)
  74. kubectl apply -n ${params.NAMESPACE} -f - <<EOF
  75. apiVersion: apps/v1
  76. kind: Deployment
  77. metadata:
  78. name: ${PROJECT_NAME}
  79. spec:
  80. replicas: 2
  81. selector:
  82. matchLabels:
  83. app: ${PROJECT_NAME}
  84. template:
  85. metadata:
  86. labels:
  87. app: ${PROJECT_NAME}
  88. spec:
  89. containers:
  90. - name: ${PROJECT_NAME}
  91. image: ${imageTag}
  92. ports:
  93. - containerPort: 8092
  94. env:
  95. - name: SPRING_PROFILES_ACTIVE
  96. value: "${params.env}"
  97. EOF
  98. # 部署 Service
  99. kubectl apply -n ${params.NAMESPACE} -f - <<EOF
  100. apiVersion: v1
  101. kind: Service
  102. metadata:
  103. name: ${PROJECT_NAME}
  104. spec:
  105. selector:
  106. app: ${PROJECT_NAME}
  107. ports:
  108. - protocol: TCP
  109. port: 80
  110. targetPort: 8092
  111. type: ClusterIP
  112. EOF
  113. # 判断是否有域名,有则部署Ingress
  114. if [ -n "${params.INGRESS_HOST}" ]; then
  115. cat <<EOT | kubectl apply -n ${params.NAMESPACE} -f -
  116. apiVersion: networking.k8s.io/v1
  117. kind: Ingress
  118. metadata:
  119. name: ${PROJECT_NAME}-ingress
  120. annotations:
  121. nginx.ingress.kubernetes.io/rewrite-target: /
  122. spec:
  123. ingressClassName: nginx
  124. rules:
  125. - host: ${params.INGRESS_HOST}
  126. http:
  127. paths:
  128. - path: /
  129. pathType: Prefix
  130. backend:
  131. service:
  132. name: ${PROJECT_NAME}
  133. port:
  134. number: 80
  135. EOT
  136. else
  137. echo ">>> 未配置 Ingress 域名,跳过 Ingress 部署"
  138. fi
  139. """
  140. echo ">>> 部署完成 ✅"
  141. }
  142. }
  143. }
  144. }
  145. post {
  146. success {
  147. echo "✅ 构建 & 部署成功 🎉"
  148. script {
  149. echo "🧹 开始清理旧镜像(只保留最新 3 个)"
  150. sh """
  151. docker images --format "{{.Repository}}:{{.Tag}} {{.CreatedAt}}" | \
  152. grep "${HARBOR_HOST}/${env.HARBOR_PROJECT}/${PROJECT_NAME}:" | \
  153. sort -rk2 | \
  154. tail -n +4 | \
  155. awk '{print \$1}' | \
  156. xargs -r docker rmi
  157. """
  158. echo "🧹 旧镜像清理完成 ✅"
  159. }
  160. }
  161. failure {
  162. echo "❌ 构建或部署失败,请检查日志!"
  163. }
  164. always {
  165. echo "🧼 清理工作区..."
  166. cleanWs()
  167. }
  168. }
  169. }