Jenkinsfile 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289
  1. pipeline {
  2. agent any
  3. parameters {
  4. choice(name: 'env', choices: ['dev', 'test', 'prod'], description: '部署环境')
  5. string(name: 'NAMESPACE', defaultValue: 'portal-frontend', description: 'K8s 命名空间')
  6. booleanParam(name: 'FORCE_UPDATE_CONFIG', defaultValue: false, description: '强制更新配置(包括 Deployment 和 Ingress)')
  7. }
  8. environment {
  9. PROJECT_NAME = 'portal-frontend'
  10. BUILD_DIR = 'dist'
  11. NODE_ENV = 'production'
  12. HARBOR_HOST = '8.130.28.21:81'
  13. KUBECONFIG_PATH = '/root/.kube/config'
  14. HARBOR_USER = 'admin'
  15. HARBOR_PASS = 'Hfln@1024'
  16. HARBOR_RETENTION_ID = '1'
  17. DOMAIN = 'radar-power.asia'
  18. TLS_SECRET = 'portal-tls'
  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. echo ">>> 域名:${env.DOMAIN}, TLS Secret:${env.TLS_SECRET}"
  28. echo ">>> 强制更新配置:${params.FORCE_UPDATE_CONFIG}"
  29. }
  30. }
  31. }
  32. stage('📥 拉取代码') {
  33. steps {
  34. checkout scm
  35. echo "✅ 代码拉取成功"
  36. }
  37. }
  38. stage('🔐 配置 Ingress 控制器') {
  39. steps {
  40. script {
  41. sh """
  42. export KUBECONFIG=${KUBECONFIG_PATH}
  43. echo ">>> 配置 Ingress 控制器为 LoadBalancer 类型..."
  44. kubectl patch svc ingress-nginx-controller -n ingress-nginx -p '{"spec":{"type":"LoadBalancer"}}' || echo "⚠️ 修改失败,可能已经是 LoadBalancer 类型"
  45. echo ">>> 当前 Ingress 控制器状态:"
  46. kubectl get svc ingress-nginx-controller -n ingress-nginx
  47. """
  48. }
  49. }
  50. }
  51. stage('🔧 构建 Docker 镜像') {
  52. steps {
  53. script {
  54. sh """
  55. docker login -u ${HARBOR_USER} -p ${HARBOR_PASS} ${HARBOR_HOST}
  56. docker build --build-arg ENV=${params.env} -t ${IMAGE_TAG} .
  57. """
  58. echo "✅ 镜像构建成功:${IMAGE_TAG}"
  59. }
  60. }
  61. }
  62. stage('🚀 推送镜像到 Harbor') {
  63. steps {
  64. script {
  65. sh """
  66. docker push ${IMAGE_TAG}
  67. docker rmi ${IMAGE_TAG}
  68. """
  69. echo "✅ 镜像推送并本地清理完成"
  70. }
  71. }
  72. }
  73. stage(' Kubernetes 部署') {
  74. steps {
  75. script {
  76. sh """
  77. export KUBECONFIG=${KUBECONFIG_PATH}
  78. # 确保命名空间存在
  79. kubectl get ns ${params.NAMESPACE} >/dev/null 2>&1 || kubectl create ns ${params.NAMESPACE}
  80. # 如果强制更新配置或 Deployment 不存在,则重新创建
  81. if [ "${params.FORCE_UPDATE_CONFIG}" = "true" ] || ! kubectl get deployment ${PROJECT_NAME} -n ${params.NAMESPACE} >/dev/null 2>&1; then
  82. if [ "${params.FORCE_UPDATE_CONFIG}" = "true" ]; then
  83. echo ">>> 强制更新配置,删除现有资源..."
  84. kubectl delete deployment ${PROJECT_NAME} -n ${params.NAMESPACE} --ignore-not-found=true
  85. kubectl delete svc ${PROJECT_NAME} -n ${params.NAMESPACE} --ignore-not-found=true
  86. kubectl delete ingress ${PROJECT_NAME}-ingress -n ${params.NAMESPACE} --ignore-not-found=true
  87. fi
  88. echo ">>> 创建 Deployment 和 Service..."
  89. kubectl apply -n ${params.NAMESPACE} -f - <<EOF
  90. apiVersion: apps/v1
  91. kind: Deployment
  92. metadata:
  93. name: ${PROJECT_NAME}
  94. spec:
  95. replicas: 2
  96. selector:
  97. matchLabels:
  98. app: ${PROJECT_NAME}
  99. template:
  100. metadata:
  101. labels:
  102. app: ${PROJECT_NAME}
  103. spec:
  104. containers:
  105. - name: ${PROJECT_NAME}
  106. image: ${IMAGE_TAG}
  107. ports:
  108. - containerPort: 80
  109. resources:
  110. requests:
  111. memory: "128Mi"
  112. cpu: "100m"
  113. limits:
  114. memory: "256Mi"
  115. cpu: "200m"
  116. livenessProbe:
  117. httpGet:
  118. path: /
  119. port: 80
  120. initialDelaySeconds: 30
  121. periodSeconds: 10
  122. readinessProbe:
  123. httpGet:
  124. path: /
  125. port: 80
  126. initialDelaySeconds: 5
  127. periodSeconds: 5
  128. ---
  129. apiVersion: v1
  130. kind: Service
  131. metadata:
  132. name: ${PROJECT_NAME}
  133. spec:
  134. type: ClusterIP
  135. selector:
  136. app: ${PROJECT_NAME}
  137. ports:
  138. - port: 80
  139. targetPort: 80
  140. protocol: TCP
  141. EOF
  142. echo ">>> 创建 Ingress..."
  143. kubectl apply -n ${params.NAMESPACE} -f - <<EOF
  144. apiVersion: networking.k8s.io/v1
  145. kind: Ingress
  146. metadata:
  147. name: ${PROJECT_NAME}-ingress
  148. annotations:
  149. kubernetes.io/ingress.class: nginx
  150. nginx.ingress.kubernetes.io/ssl-redirect: "true"
  151. nginx.ingress.kubernetes.io/force-ssl-redirect: "true"
  152. nginx.ingress.kubernetes.io/backend-protocol: "HTTP"
  153. nginx.ingress.kubernetes.io/ssl-passthrough: "false"
  154. nginx.ingress.kubernetes.io/proxy-body-size: "8m"
  155. nginx.ingress.kubernetes.io/rewrite-target: /
  156. nginx.ingress.kubernetes.io/ssl-ciphers: "ECDHE-RSA-AES128-GCM-SHA256:ECDHE-RSA-AES256-GCM-SHA384:ECDHE-RSA-AES128-SHA256:ECDHE-RSA-AES256-SHA384"
  157. nginx.ingress.kubernetes.io/ssl-protocols: "TLSv1.2 TLSv1.3"
  158. spec:
  159. tls:
  160. - hosts:
  161. - ${env.DOMAIN}
  162. secretName: ${env.TLS_SECRET}
  163. rules:
  164. - host: ${env.DOMAIN}
  165. http:
  166. paths:
  167. - path: /
  168. pathType: Prefix
  169. backend:
  170. service:
  171. name: ${PROJECT_NAME}
  172. port:
  173. number: 80
  174. EOF
  175. else
  176. # 只更新镜像
  177. echo ">>> Deployment 已存在,仅更新镜像..."
  178. kubectl set image deployment/${PROJECT_NAME} ${PROJECT_NAME}=${IMAGE_TAG} -n ${params.NAMESPACE}
  179. echo "✅ 镜像更新完成"
  180. fi
  181. # 显示部署状态
  182. echo ">>> 部署状态:"
  183. kubectl get all -n ${params.NAMESPACE}
  184. kubectl get ingress -n ${params.NAMESPACE}
  185. echo "✅ 应用部署完成!"
  186. echo "�� 访问地址:https://${env.DOMAIN}"
  187. echo " 注意:请确保域名 ${env.DOMAIN} 已正确解析到集群"
  188. """
  189. }
  190. }
  191. }
  192. stage('🔍 部署验证') {
  193. steps {
  194. script {
  195. sh """
  196. export KUBECONFIG=${KUBECONFIG_PATH}
  197. echo ">>> 验证部署状态..."
  198. # 检查 Pod 状态
  199. kubectl get pods -n ${params.NAMESPACE} -l app=${PROJECT_NAME}
  200. # 检查 Service 状态
  201. kubectl get svc -n ${params.NAMESPACE} ${PROJECT_NAME}
  202. # 检查 Ingress 状态
  203. kubectl get ingress -n ${params.NAMESPACE} ${PROJECT_NAME}-ingress
  204. # 检查 TLS Secret
  205. kubectl get secret -n ${params.NAMESPACE} ${env.TLS_SECRET}
  206. echo "✅ 部署验证完成"
  207. """
  208. }
  209. }
  210. }
  211. stage('🧹 清理本地旧镜像(保留最新3个)') {
  212. steps {
  213. script {
  214. def baseImage = "${HARBOR_HOST}/${env.HARBOR_PROJECT}/${PROJECT_NAME}"
  215. sh """
  216. docker images ${baseImage} --format "{{.Repository}}:{{.Tag}}" \\
  217. | grep -v latest \\
  218. | sort -r -t ':' -k2 \\
  219. | tail -n +4 \\
  220. | xargs -r docker rmi || true
  221. """
  222. echo "✅ 本地旧镜像清理完成"
  223. }
  224. }
  225. }
  226. stage(' 清理悬空镜像 <none>') {
  227. steps {
  228. script {
  229. sh """
  230. docker images -f "dangling=true" -q | xargs -r docker rmi || true
  231. """
  232. echo "✅ 悬空镜像(<none>)清理完成"
  233. }
  234. }
  235. }
  236. stage(' 触发 Harbor 镜像保留策略') {
  237. steps {
  238. script {
  239. sh """
  240. curl -u ${HARBOR_USER}:${HARBOR_PASS} -X POST \\
  241. "http://${HARBOR_HOST}/api/v2.0/retentions/${HARBOR_RETENTION_ID}/executions"
  242. """
  243. echo "✅ Harbor 镜像保留策略已触发"
  244. }
  245. }
  246. }
  247. }
  248. post {
  249. success {
  250. echo "✅ 构建 & 部署成功!"
  251. echo "🌐 应用可通过 https://${env.DOMAIN} 访问"
  252. }
  253. failure {
  254. echo "❌ 构建或部署失败,请检查日志"
  255. }
  256. always {
  257. cleanWs()
  258. }
  259. }
  260. }