Jenkinsfile 6.7 KB

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