Jenkinsfile 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. pipeline {
  2. agent any
  3. parameters {
  4. choice(name: 'env', choices: ['dev', 'test', 'prod'], description: '部署环境')
  5. string(name: 'NAMESPACE', defaultValue: 'hfln-dev', description: 'K8s 命名空间')
  6. }
  7. environment {
  8. PROJECT_NAME = 'portal-service-frontend'
  9. BUILD_DIR = 'dist'
  10. NODE_ENV = 'production'
  11. HARBOR_HOST = '8.130.28.21:81'
  12. KUBECONFIG_PATH = '/root/.kube/config'
  13. NODE1_IP = '172.27.73.147'
  14. NODE2_IP = '172.27.73.146'
  15. HARBOR_USER = 'admin'
  16. HARBOR_PASS = 'Hfln@1024'
  17. HARBOR_RETENTION_ID = '1'
  18. DOMAIN = 'radar-power.asia'
  19. }
  20. stages {
  21. stage('🧬 初始化环境') {
  22. steps {
  23. script {
  24. env.HARBOR_PROJECT = params.env
  25. env.IMAGE_TAG = "${env.HARBOR_HOST}/${env.HARBOR_PROJECT}/${env.PROJECT_NAME}:${BUILD_NUMBER}"
  26. echo ">>> 环境:${params.env}, Harbor项目:${env.HARBOR_PROJECT}, K8s命名空间:${params.NAMESPACE}"
  27. echo ">>> IMAGE_TAG = ${env.IMAGE_TAG}"
  28. echo ">>> 域名:https://${env.DOMAIN}/"
  29. }
  30. }
  31. }
  32. stage('📥 拉取代码') {
  33. steps {
  34. checkout scm
  35. echo "✅ 代码拉取成功"
  36. }
  37. }
  38. stage('🔧 构建 Docker 镜像') {
  39. steps {
  40. script {
  41. sh """
  42. docker login -u ${env.HARBOR_USER} -p ${env.HARBOR_PASS} ${env.HARBOR_HOST}
  43. docker build --build-arg ENV=${params.env} -t ${env.IMAGE_TAG} .
  44. """
  45. echo "✅ 镜像构建成功:${env.IMAGE_TAG}"
  46. }
  47. }
  48. }
  49. stage('🚀 推送镜像到 Harbor') {
  50. steps {
  51. script {
  52. sh """
  53. docker push ${env.IMAGE_TAG}
  54. docker rmi ${env.IMAGE_TAG} || true
  55. """
  56. echo "✅ 镜像推送并本地清理完成"
  57. }
  58. }
  59. }
  60. stage('🔍 测试节点能否拉取镜像') {
  61. steps {
  62. script {
  63. echo ">>> 测试节点能否拉取镜像..."
  64. sh """
  65. ssh root@${env.NODE1_IP} docker login -u ${env.HARBOR_USER} -p ${env.HARBOR_PASS} ${env.HARBOR_HOST} && docker pull ${env.IMAGE_TAG} || echo '[❌ 节点 ${env.NODE1_IP} 拉取失败]'
  66. ssh root@${env.NODE2_IP} docker login -u ${env.HARBOR_USER} -p ${env.HARBOR_PASS} ${env.HARBOR_HOST} && docker pull ${env.IMAGE_TAG} || echo '[❌ 节点 ${env.NODE2_IP} 拉取失败]'
  67. """
  68. }
  69. }
  70. }
  71. stage('📦 部署到 Kubernetes(带 Ingress)') {
  72. steps {
  73. script {
  74. def ingressBlock = """---
  75. apiVersion: networking.k8s.io/v1
  76. kind: Ingress
  77. metadata:
  78. name: ${env.PROJECT_NAME}
  79. namespace: ${params.NAMESPACE}
  80. annotations:
  81. nginx.ingress.kubernetes.io/rewrite-target: /
  82. spec:
  83. tls:
  84. - hosts:
  85. - ${env.DOMAIN}
  86. secretName: portal-tls
  87. rules:
  88. - host: ${env.DOMAIN}
  89. http:
  90. paths:
  91. - path: /
  92. pathType: Prefix
  93. backend:
  94. service:
  95. name: ${env.PROJECT_NAME}
  96. port:
  97. number: 80
  98. """
  99. def deployYaml = """apiVersion: apps/v1
  100. kind: Deployment
  101. metadata:
  102. name: ${env.PROJECT_NAME}
  103. namespace: ${params.NAMESPACE}
  104. spec:
  105. replicas: 2
  106. selector:
  107. matchLabels:
  108. app: ${env.PROJECT_NAME}
  109. template:
  110. metadata:
  111. labels:
  112. app: ${env.PROJECT_NAME}
  113. spec:
  114. containers:
  115. - name: ${env.PROJECT_NAME}
  116. image: ${env.IMAGE_TAG}
  117. ports:
  118. - containerPort: 80
  119. env:
  120. - name: NODE_ENV
  121. value: "${params.env}"
  122. ---
  123. apiVersion: v1
  124. kind: Service
  125. metadata:
  126. name: ${env.PROJECT_NAME}
  127. namespace: ${params.NAMESPACE}
  128. spec:
  129. type: NodePort
  130. selector:
  131. app: ${env.PROJECT_NAME}
  132. ports:
  133. - port: 80
  134. targetPort: 80
  135. nodePort: 30088
  136. ${ingressBlock}
  137. """
  138. writeFile file: 'deploy.yaml', text: deployYaml
  139. sh """
  140. export KUBECONFIG=${env.KUBECONFIG_PATH}
  141. kubectl get ns ${params.NAMESPACE} >/dev/null 2>&1 || kubectl create ns ${params.NAMESPACE}
  142. kubectl apply -f deploy.yaml
  143. kubectl rollout status deployment/${env.PROJECT_NAME} -n ${params.NAMESPACE} --timeout=120s || echo '[rollout timeout or incomplete]'
  144. """
  145. echo ">>> ✅ 部署完成,访问地址:https://${env.DOMAIN}/ (请确保 DNS 已指向 Ingress 公网 IP 且 secret portal-tls 已创建)"
  146. }
  147. }
  148. }
  149. stage('🧹 清理本地旧镜像(保留最新3个)') {
  150. steps {
  151. script {
  152. def baseImage = "${env.HARBOR_HOST}/${env.HARBOR_PROJECT}/${env.PROJECT_NAME}"
  153. sh """
  154. docker images ${baseImage} --format "{{.Repository}}:{{.Tag}}" \\
  155. | grep -v latest \\
  156. | sort -r -t ':' -k2 \\
  157. | tail -n +4 \\
  158. | xargs -r docker rmi || true
  159. """
  160. echo "✅ 本地旧镜像清理完成"
  161. }
  162. }
  163. }
  164. stage('🧼 清理 dangling 镜像') {
  165. steps {
  166. script {
  167. sh """
  168. docker images -f "dangling=true" -q | xargs -r docker rmi || true
  169. """
  170. echo "✅ 悬空镜像(<none>)清理完成"
  171. }
  172. }
  173. }
  174. stage('🔁 触发 Harbor 镜像保留策略(可选)') {
  175. steps {
  176. script {
  177. sh """
  178. curl -u ${env.HARBOR_USER}:${env.HARBOR_PASS} -X POST "http://${env.HARBOR_HOST}/api/v2.0/retentions/${env.HARBOR_RETENTION_ID}/executions" || echo '[retention trigger failed]'
  179. """
  180. echo "✅ Harbor 镜像保留策略已触发(若配置)"
  181. }
  182. }
  183. }
  184. }
  185. post {
  186. success {
  187. echo "✅ 构建 & 部署成功 🎉"
  188. }
  189. failure {
  190. echo "❌ 构建或部署失败,请检查日志"
  191. }
  192. always {
  193. cleanWs()
  194. }
  195. }
  196. }