Jenkinsfile 5.6 KB

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