Jenkinsfile 11 KB

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