Jenkinsfile 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257
  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. string(name: 'DOMAIN', defaultValue: 'radar-power.asia', description: 'Ingress 域名(留空则不创建 Ingress)')
  7. string(name: 'TLS_SECRET', defaultValue: 'portal-tls', description: 'TLS Secret 名称(仅在 DOMAIN 非空时使用)')
  8. }
  9. environment {
  10. PROJECT_NAME = 'new-portal-frontend'
  11. BUILD_DIR = 'dist'
  12. NODE_ENV = 'production'
  13. HARBOR_HOST = '8.130.28.21:81'
  14. KUBECONFIG_PATH = '/root/.kube/config'
  15. HARBOR_USER = 'admin'
  16. HARBOR_PASS = 'Hfln@1024'
  17. HARBOR_RETENTION_ID = '1'
  18. }
  19. stages {
  20. stage('🧬 初始化环境') {
  21. steps {
  22. script {
  23. env.HARBOR_PROJECT = params.env
  24. env.IMAGE_TAG = "${HARBOR_HOST}/${env.HARBOR_PROJECT}/${PROJECT_NAME}:${BUILD_NUMBER}"
  25. echo ">>> 环境:${params.env}, Harbor项目:${env.HARBOR_PROJECT}, K8s命名空间:${params.NAMESPACE}"
  26. if (params.DOMAIN?.trim()) {
  27. echo ">>> 域名:${params.DOMAIN}, TLS Secret:${params.TLS_SECRET}"
  28. }
  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 ${HARBOR_USER} -p ${HARBOR_PASS} ${HARBOR_HOST}
  43. docker build --build-arg ENV=${params.env} -t ${IMAGE_TAG} .
  44. """
  45. echo "✅ 镜像构建成功:${IMAGE_TAG}"
  46. }
  47. }
  48. }
  49. stage('🚀 推送镜像到 Harbor') {
  50. steps {
  51. script {
  52. sh """
  53. docker push ${IMAGE_TAG}
  54. docker rmi ${IMAGE_TAG}
  55. """
  56. echo "✅ 镜像推送并本地清理完成"
  57. }
  58. }
  59. }
  60. stage(' Kubernetes 部署') {
  61. steps {
  62. script {
  63. def domain = params.DOMAIN?.trim()
  64. sh """
  65. export KUBECONFIG=${KUBECONFIG_PATH}
  66. # 确保命名空间存在
  67. kubectl get ns ${params.NAMESPACE} >/dev/null 2>&1 || kubectl create ns ${params.NAMESPACE}
  68. # 如果 Deployment 存在,更新镜像
  69. if kubectl get deployment ${PROJECT_NAME} -n ${params.NAMESPACE} >/dev/null 2>&1; then
  70. echo ">>> Deployment 已存在,更新镜像..."
  71. kubectl set image deployment/${PROJECT_NAME} ${PROJECT_NAME}=${IMAGE_TAG} -n ${params.NAMESPACE}
  72. kubectl rollout status deployment/${PROJECT_NAME} -n ${params.NAMESPACE} --timeout=120s
  73. # 更新 Ingress(如果提供了域名)
  74. if [ -n "${domain}" ]; then
  75. kubectl apply -n ${params.NAMESPACE} -f - <<EOF
  76. apiVersion: networking.k8s.io/v1
  77. kind: Ingress
  78. metadata:
  79. name: ${PROJECT_NAME}-ingress
  80. annotations:
  81. kubernetes.io/ingress.class: nginx
  82. nginx.ingress.kubernetes.io/ssl-redirect: "true"
  83. nginx.ingress.kubernetes.io/force-ssl-redirect: "true"
  84. nginx.ingress.kubernetes.io/backend-protocol: "HTTP"
  85. nginx.ingress.kubernetes.io/ssl-passthrough: "false"
  86. nginx.ingress.kubernetes.io/proxy-body-size: "8m"
  87. nginx.ingress.kubernetes.io/rewrite-target: /
  88. spec:
  89. tls:
  90. - hosts:
  91. - ${domain}
  92. secretName: ${params.TLS_SECRET}
  93. rules:
  94. - host: ${domain}
  95. http:
  96. paths:
  97. - path: /
  98. pathType: Prefix
  99. backend:
  100. service:
  101. name: ${PROJECT_NAME}
  102. port:
  103. number: 80
  104. EOF
  105. fi
  106. else
  107. # 创建新的 Deployment 和 Service
  108. kubectl apply -n ${params.NAMESPACE} -f - <<EOF
  109. apiVersion: apps/v1
  110. kind: Deployment
  111. metadata:
  112. name: ${PROJECT_NAME}
  113. spec:
  114. replicas: 2
  115. selector:
  116. matchLabels:
  117. app: ${PROJECT_NAME}
  118. template:
  119. metadata:
  120. labels:
  121. app: ${PROJECT_NAME}
  122. spec:
  123. containers:
  124. - name: ${PROJECT_NAME}
  125. image: ${IMAGE_TAG}
  126. ports:
  127. - containerPort: 80
  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. EOF
  141. # 创建 Ingress(如果提供了域名)
  142. if [ -n "${domain}" ]; then
  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. spec:
  157. tls:
  158. - hosts:
  159. - ${domain}
  160. secretName: ${params.TLS_SECRET}
  161. rules:
  162. - host: ${domain}
  163. http:
  164. paths:
  165. - path: /
  166. pathType: Prefix
  167. backend:
  168. service:
  169. name: ${PROJECT_NAME}
  170. port:
  171. number: 80
  172. EOF
  173. fi
  174. fi
  175. # 显示部署状态
  176. echo ">>> 部署状态:"
  177. kubectl get all -n ${params.NAMESPACE}
  178. kubectl get ingress -n ${params.NAMESPACE} || echo ">>> 未配置 Ingress"
  179. # 如果配置了域名,显示访问信息
  180. if [ -n "${domain}" ]; then
  181. echo ">>> 应用部署完成!"
  182. echo ">>> 访问地址:https://${domain}"
  183. echo ">>> 注意:确保域名 ${domain} 已正确解析到集群"
  184. else
  185. echo ">>> 应用部署完成!"
  186. echo ">>> 注意:未配置域名,请手动配置 Ingress 或使用 NodePort 访问"
  187. fi
  188. """
  189. }
  190. }
  191. }
  192. stage('🧹 清理本地旧镜像(保留最新3个)') {
  193. steps {
  194. script {
  195. def baseImage = "${HARBOR_HOST}/${env.HARBOR_PROJECT}/${PROJECT_NAME}"
  196. sh """
  197. docker images ${baseImage} --format "{{.Repository}}:{{.Tag}}" \\
  198. | grep -v latest \\
  199. | sort -r -t ':' -k2 \\
  200. | tail -n +4 \\
  201. | xargs -r docker rmi || true
  202. """
  203. echo "✅ 本地旧镜像清理完成"
  204. }
  205. }
  206. }
  207. stage(' 清理悬空镜像 <none>') {
  208. steps {
  209. script {
  210. sh """
  211. docker images -f "dangling=true" -q | xargs -r docker rmi || true
  212. """
  213. echo "✅ 悬空镜像(<none>)清理完成"
  214. }
  215. }
  216. }
  217. stage(' 触发 Harbor 镜像保留策略') {
  218. steps {
  219. script {
  220. sh """
  221. curl -u ${HARBOR_USER}:${HARBOR_PASS} -X POST \\
  222. "http://${HARBOR_HOST}/api/v2.0/retentions/${HARBOR_RETENTION_ID}/executions"
  223. """
  224. echo "✅ Harbor 镜像保留策略已触发"
  225. }
  226. }
  227. }
  228. }
  229. post {
  230. success {
  231. echo "✅ 构建 & 部署成功 "
  232. }
  233. failure {
  234. echo "❌ 构建或部署失败,请检查日志"
  235. }
  236. always {
  237. cleanWs()
  238. }
  239. }
  240. }