Jenkinsfile 8.1 KB

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