Jenkinsfile 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. pipeline {
  2. agent any
  3. options {
  4. buildDiscarder(logRotator(numToKeepStr: '5'))
  5. timestamps()
  6. }
  7. parameters {
  8. string(name: 'BRANCH', defaultValue: 'uat', description: '要部署到UAT的分支')
  9. }
  10. environment {
  11. // Dockerfile 路径(相对于代码根目录,即 Jenkins workspace)
  12. DOCKERFILE_STORE = "alien_store/Dockerfile"
  13. DOCKERFILE_GATEWAY = "alien_gateway/Dockerfile"
  14. DOCKERFILE_CONTRACT = "alien_contract/Dockerfile"
  15. // 如需推送远端 Registry,填写地址,否则留空
  16. REGISTRY = ""
  17. REGISTRY_CRED = "registry-cred-id"
  18. // 镜像标签统一使用 uat
  19. IMAGE_STORE = "${REGISTRY ? REGISTRY + '/alien_store:uat-' + env.BUILD_NUMBER : 'alien_store:uat'}"
  20. IMAGE_GATEWAY = "${REGISTRY ? REGISTRY + '/alien_gateway:uat-' + env.BUILD_NUMBER : 'alien_gateway:uat'}"
  21. IMAGE_CONTRACT = "${REGISTRY ? REGISTRY + '/alien_contract:uat-' + env.BUILD_NUMBER : 'alien_contract:uat'}"
  22. // 容器名(store 不要再用 esign-uat,会和真正的 e签宝容器撞名)
  23. CONTAINER_NAME_STORE = "alien_store_py-uat"
  24. CONTAINER_NAME_GATEWAY = "alien_gateway_py-uat"
  25. CONTAINER_NAME_CONTRACT = "alien_contract_py-uat"
  26. // 容器内端口(必须与各服务 Dockerfile 中 uvicorn 监听端口一致)
  27. PORT_STORE = "8001"
  28. PORT_GATEWAY = "43333"
  29. PORT_CONTRACT = "8002"
  30. // 使用独立的 UAT 网络
  31. DOCKER_NET = "alien_net_uat"
  32. // UAT 日志目录
  33. LOG_ROOT = "/docker/python-uat/logs"
  34. }
  35. stages {
  36. // 注意:使用 "Pipeline script from SCM" 模式时,
  37. // Jenkins 会自动 checkout 代码到 workspace,不需要在 Jenkinsfile 里再写 Checkout stage。
  38. stage('Build Images') {
  39. steps {
  40. script {
  41. // 直接在当前 workspace(Jenkins 已 checkout 的代码根)里构建
  42. if (env.REGISTRY?.trim()) {
  43. withDockerRegistry(credentialsId: env.REGISTRY_CRED, url: "") {
  44. sh "docker build -f ${DOCKERFILE_STORE} -t ${IMAGE_STORE} ."
  45. sh "docker build -f ${DOCKERFILE_GATEWAY} -t ${IMAGE_GATEWAY} ."
  46. sh "docker build -f ${DOCKERFILE_CONTRACT} -t ${IMAGE_CONTRACT} ."
  47. }
  48. } else {
  49. sh "docker build -f ${DOCKERFILE_STORE} -t ${IMAGE_STORE} ."
  50. sh "docker build -f ${DOCKERFILE_GATEWAY} -t ${IMAGE_GATEWAY} ."
  51. sh "docker build -f ${DOCKERFILE_CONTRACT} -t ${IMAGE_CONTRACT} ."
  52. }
  53. }
  54. }
  55. }
  56. stage('Push Images') {
  57. when { expression { return env.REGISTRY?.trim() as boolean } }
  58. steps {
  59. withDockerRegistry(credentialsId: env.REGISTRY_CRED, url: "") {
  60. sh "docker push ${IMAGE_STORE}"
  61. sh "docker push ${IMAGE_GATEWAY}"
  62. sh "docker push ${IMAGE_CONTRACT}"
  63. }
  64. }
  65. }
  66. stage('Deploy') {
  67. steps {
  68. script {
  69. echo ">>> UAT 部署镜像: ${IMAGE_STORE} / ${IMAGE_GATEWAY} / ${IMAGE_CONTRACT}"
  70. sh """
  71. # 创建 UAT 专用网络
  72. docker network create ${DOCKER_NET} 2>/dev/null || true
  73. # 创建日志目录
  74. mkdir -p ${LOG_ROOT}/store ${LOG_ROOT}/gateway ${LOG_ROOT}/contract
  75. # 把当前 workspace 的 .env.uat 复制到宿主机持久目录,供容器挂载使用
  76. mkdir -p /docker/python-uat
  77. cp .env.uat /docker/python-uat/.env.uat
  78. # 停止旧容器(不会误删 java 的 gateway-uat / store-uat / 真正的 esign-uat 等)
  79. docker rm -f ${CONTAINER_NAME_STORE} ${CONTAINER_NAME_GATEWAY} ${CONTAINER_NAME_CONTRACT} 2>/dev/null || true
  80. # 1) 先启动下游:store
  81. docker run -d --name ${CONTAINER_NAME_STORE} \\
  82. --network ${DOCKER_NET} \\
  83. -e APP_ENV=uat \\
  84. -v /docker/python-uat/.env.uat:/app/.env.uat:ro \\
  85. -v ${LOG_ROOT}/store:/app/common/logs/alien_store \\
  86. --restart unless-stopped \\
  87. ${IMAGE_STORE}
  88. # 2) 再启动下游:contract
  89. docker run -d --name ${CONTAINER_NAME_CONTRACT} \\
  90. --network ${DOCKER_NET} \\
  91. -e APP_ENV=uat \\
  92. -v /docker/python-uat/.env.uat:/app/.env.uat:ro \\
  93. -v ${LOG_ROOT}/contract:/app/common/logs/alien_contract \\
  94. --restart unless-stopped \\
  95. ${IMAGE_CONTRACT}
  96. # 3) 最后启动网关:gateway(依赖上面两个下游;唯一对外暴露的入口)
  97. # STORE_BASE_URL / CONTRACT_BASE_URL 通过 -e 覆盖 .env.uat 中的本地默认值
  98. docker run -d --name ${CONTAINER_NAME_GATEWAY} \\
  99. --network ${DOCKER_NET} \\
  100. -p ${PORT_GATEWAY}:${PORT_GATEWAY} \\
  101. -e APP_ENV=uat \\
  102. -e STORE_BASE_URL=http://${CONTAINER_NAME_STORE}:${PORT_STORE} \\
  103. -e CONTRACT_BASE_URL=http://${CONTAINER_NAME_CONTRACT}:${PORT_CONTRACT} \\
  104. -v /docker/python-uat/.env.uat:/app/.env.uat:ro \\
  105. -v ${LOG_ROOT}/gateway:/app/common/logs/alien_gateway \\
  106. --restart unless-stopped \\
  107. ${IMAGE_GATEWAY}
  108. """
  109. }
  110. }
  111. }
  112. }
  113. post {
  114. success {
  115. echo "UAT 环境部署成功!"
  116. }
  117. failure {
  118. echo "UAT 环境部署失败,请检查日志。"
  119. }
  120. always {
  121. cleanWs()
  122. }
  123. }
  124. }