From 35acf595e415840aa6bcfbc3dfb140a827eb9453 Mon Sep 17 00:00:00 2001 From: zk Date: Thu, 21 May 2026 18:06:19 +0800 Subject: [PATCH] =?UTF-8?q?=E6=B7=BB=E5=8A=A0=E9=83=A8=E7=BD=B2=E6=96=87?= =?UTF-8?q?=E4=BB=B6?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- client-api/Dockerfile | 47 ++++ client-api/Jenkinsfile | 206 ++++++++++++++++++ client-api/nginx.conf | 55 +++++ ...lication-local.yml => application-dev.yml} | 0 .../src/main/resources/application-prod.yml | 118 ++++++++++ .../src/main/resources/application-test.yml | 118 ++++++++++ client-api/src/main/resources/application.yml | 2 +- docker-compose.client-api.yml | 82 +++++++ settings.xml | 14 ++ 9 files changed, 641 insertions(+), 1 deletion(-) create mode 100644 client-api/Dockerfile create mode 100644 client-api/Jenkinsfile create mode 100644 client-api/nginx.conf rename client-api/src/main/resources/{application-local.yml => application-dev.yml} (100%) create mode 100644 client-api/src/main/resources/application-prod.yml create mode 100644 client-api/src/main/resources/application-test.yml create mode 100644 docker-compose.client-api.yml create mode 100644 settings.xml diff --git a/client-api/Dockerfile b/client-api/Dockerfile new file mode 100644 index 0000000..8f172b4 --- /dev/null +++ b/client-api/Dockerfile @@ -0,0 +1,47 @@ +# syntax=docker/dockerfile:1 + +# ==================== 第一阶段:编译 ==================== +FROM maven:3.8-openjdk-17 AS builder +WORKDIR /build + +# 阿里云 Maven 镜像加速 +COPY settings.xml /root/.m2/settings.xml + +# 先拷贝 pom 利用 Docker 缓存层(依赖不变时跳过下载) +COPY pom.xml . +COPY common/pom.xml common/pom.xml +COPY manager/pom.xml manager/pom.xml +COPY client-api/pom.xml client-api/pom.xml + +# 拷贝源码 +COPY . . + +# 构建(从父 pom 编译,client-api 依赖 common 和 manager) +RUN --mount=type=cache,target=/root/.m2/repository \ + mvn clean package -pl client-api -am -DskipTests + +# ==================== 第二阶段:运行 ==================== +FROM alibabadragonwell/dragonwell:17-anolis + +ENV TZ=Asia/Shanghai +ENV PROFILES_ACTIVE=prod +ENV JAVA_OPTS="-XX:MaxRAMPercentage=75.0 \ + -XX:+UseG1GC \ + -XX:+HeapDumpOnOutOfMemoryError \ + -XX:HeapDumpPath=/app/logs/heapdump.hprof \ + -Dfile.encoding=UTF-8 \ + -Duser.timezone=Asia/Shanghai" + +# 时区 + curl(健康检查用) +RUN ln -sf /usr/share/zoneinfo/$TZ /etc/localtime && echo $TZ > /etc/timezone \ + && yum install -y curl && yum clean all + +RUN mkdir -p /app/logs + +WORKDIR /app + +EXPOSE 8080 + +COPY --from=builder /build/client-api/target/client-api.jar ./client-api.jar + +CMD java $JAVA_OPTS -jar client-api.jar diff --git a/client-api/Jenkinsfile b/client-api/Jenkinsfile new file mode 100644 index 0000000..2399606 --- /dev/null +++ b/client-api/Jenkinsfile @@ -0,0 +1,206 @@ +/** + * OfferPie Backend Client-API 蓝绿部署流水线 + * + * 工作目录说明: + * - Jenkins 会自动为每个项目创建独立的工作空间:/var/jenkins_home/workspace/<项目名>/ + * - docker-compose 文件在项目根目录,通过 -f 指定文件名 + * - Dockerfile 在 client-api/ 子目录下,build context 为项目根目录(因为需要编译 common 和 manager 模块) + * - nginx.conf 在 client-api/ 下,首次部署时 docker cp 进 nginx 容器 + */ +pipeline { + agent any + + parameters { + choice(name: 'BRANCH', choices: ['master', 'pre', 'dev', 'test'], description: '选择要部署的分支') + } + + environment { + COMPOSE_FILE = 'docker-compose.client-api.yml' // docker-compose 文件名 + CONTAINER_PREFIX = 'offerpie-backend-client' // 容器名前缀,拼接 -blue / -green / -nginx + HEALTH_URL = 'http://localhost:8080/api/public/actuator/health' // 应用健康检查地址 + } + + stages { + stage('开始提示') { + steps { + echo "OfferPie Backend Client-API 开始构建" + } + } + + stage('拉取代码') { + steps { + echo "拉取 ${params.BRANCH} 分支代码" + git branch: "${params.BRANCH}", + credentialsId: 'ef5fffc1-9b35-403d-9ca6-e1b73eb0e45a', + url: 'https://codeup.aliyun.com/5f0ed3b9769820a3e817dee2/offerpie/offerpie_backend.git' + } + } + + /** + * 检测部署目标 + * - 检查 blue 和 green 容器的运行状态 + * - blue 在运行 → 部署 green;green 在运行 → 部署 blue + * - 都不在或都在 → 默认部署 blue + */ + stage('检测部署目标') { + steps { + echo "检查当前容器状态" + script { + def blueRunning = sh(script: "docker ps -q -f name=${CONTAINER_PREFIX}-blue", returnStdout: true).trim() + def greenRunning = sh(script: "docker ps -q -f name=${CONTAINER_PREFIX}-green", returnStdout: true).trim() + + env.DEPLOY_TARGET = '' + + if (blueRunning && !greenRunning) { + env.DEPLOY_TARGET = 'green' + } + if (greenRunning && !blueRunning) { + env.DEPLOY_TARGET = 'blue' + } + if (!env.DEPLOY_TARGET) { + echo "当前环境未部署服务或状态异常,默认部署 blue" + env.DEPLOY_TARGET = 'blue' + } + + env.OTHER_TARGET = (env.DEPLOY_TARGET == 'blue') ? 'green' : 'blue' + echo "当前激活: ${env.OTHER_TARGET},即将部署: ${env.DEPLOY_TARGET}" + } + } + } + + /** + * 构建新版本 + * - 先清理目标颜色的旧容器(如果残留) + * - docker-compose build 编译新镜像 + */ + stage('构建新版本') { + steps { + script { + def existingContainer = sh(script: "docker ps -aq -f name=${CONTAINER_PREFIX}-${env.DEPLOY_TARGET}", returnStdout: true).trim() + if (existingContainer) { + echo "清理已存在的容器: ${CONTAINER_PREFIX}-${env.DEPLOY_TARGET}" + sh "docker rm -f ${CONTAINER_PREFIX}-${env.DEPLOY_TARGET}" + } + sh "docker-compose -f ${COMPOSE_FILE} build ${env.DEPLOY_TARGET}" + } + } + } + + /** + * 检查 Nginx + * - 检查 nginx 容器是否存在 + * - 不存在则启动并复制配置文件(首次部署) + * - 存在则确保容器运行中 + */ + stage('检查Nginx') { + steps { + script { + def nginxExists = sh(script: "docker ps -aq -f name=${CONTAINER_PREFIX}-nginx", returnStdout: true).trim() + if (!nginxExists) { + echo "首次部署,初始化 Nginx 容器" + sh "docker-compose -f ${COMPOSE_FILE} up -d nginx" + sh "sleep 3" + // 复制配置文件到容器内 + sh "docker cp client-api/nginx.conf ${CONTAINER_PREFIX}-nginx:/etc/nginx/nginx.conf" + sh "docker exec ${CONTAINER_PREFIX}-nginx nginx -s reload" + } else { + def nginxRunning = sh(script: "docker ps -q -f name=${CONTAINER_PREFIX}-nginx", returnStdout: true).trim() + if (!nginxRunning) { + echo "Nginx 容器已停止,重新启动" + sh "docker start ${CONTAINER_PREFIX}-nginx" + sh "sleep 2" + } + echo "Nginx 容器已存在且运行中" + } + } + } + } + + stage('启动新版本') { + steps { + script { + sh "docker-compose -f ${COMPOSE_FILE} up -d ${env.DEPLOY_TARGET}" + // 等待 Spring Boot 应用完全启动 + sh 'sleep 35' + } + } + } + + /** + * 健康检查 + * - 进入容器执行 curl 检测应用是否正常响应 + * - curl -f 参数:HTTP 错误码(404、500等)时返回失败 + * - 最多重试 3 次,每次间隔 5 秒 + * - 全部失败则终止部署流程 + */ + stage('健康检查') { + steps { + script { + def maxRetries = 3 + def retryCount = 0 + def healthy = false + + while (retryCount < maxRetries && !healthy) { + try { + sh "docker exec ${CONTAINER_PREFIX}-${env.DEPLOY_TARGET} curl -f ${HEALTH_URL}" + healthy = true + } catch (Exception e) { + retryCount++ + if (retryCount < maxRetries) { + echo "健康检查失败,5秒后重试 (${retryCount}/${maxRetries})" + sleep 5 + } + } + } + + if (!healthy) { + error "健康检查失败,部署中止" + } + echo "✅ ${CONTAINER_PREFIX}-${env.DEPLOY_TARGET} 健康检查通过" + } + } + } + + /** + * 切换流量 + * - 修改 nginx 配置中的 proxy_pass 指向新版本容器 + * - nginx 不挂载宿主机文件,直接 sed -i 修改容器内配置 + * - reload 使新配置生效,实现零停机切换 + */ + stage('切换流量') { + steps { + script { + sh "docker exec ${CONTAINER_PREFIX}-nginx sed -i 's/proxy_pass http:\\/\\/\\(blue\\|green\\):8080;/proxy_pass http:\\/\\/${env.DEPLOY_TARGET}:8080;/' /etc/nginx/nginx.conf" + sh "docker exec ${CONTAINER_PREFIX}-nginx nginx -s reload" + echo "✅ 流量已切换到 ${env.DEPLOY_TARGET}" + } + } + } + + /** + * 停止并删除旧版本 + * - 不能用 docker-compose down,会删除所有服务(包括 nginx) + * - 用 docker rm -f 只删除指定的旧颜色容器 + */ + stage('删除旧版本') { + steps { + script { + def otherExists = sh(script: "docker ps -aq -f name=${CONTAINER_PREFIX}-${env.OTHER_TARGET}", returnStdout: true).trim() + if (otherExists) { + sh "docker rm -f ${CONTAINER_PREFIX}-${env.OTHER_TARGET}" + echo "旧版本 ${env.OTHER_TARGET} 已删除" + } + } + } + } + } + + post { + success { + echo "✅ 蓝绿部署成功!当前运行: ${env.DEPLOY_TARGET}" + } + failure { + echo '❌ 部署失败,请检查日志' + } + } +} diff --git a/client-api/nginx.conf b/client-api/nginx.conf new file mode 100644 index 0000000..f64f280 --- /dev/null +++ b/client-api/nginx.conf @@ -0,0 +1,55 @@ +worker_processes auto; +error_log /var/log/nginx/error.log warn; +pid /var/run/nginx.pid; + +events { + worker_connections 1024; + use epoll; + multi_accept on; +} + +http { + include /etc/nginx/mime.types; + default_type application/octet-stream; + + log_format main '$remote_addr - $remote_user [$time_local] "$request" ' + '$status $body_bytes_sent "$http_referer" ' + '"$http_user_agent" "$http_x_forwarded_for"'; + access_log /var/log/nginx/access.log main; + + sendfile on; + tcp_nopush on; + tcp_nodelay on; + keepalive_timeout 65; + + client_max_body_size 20m; + + server { + listen 80; + server_name _; + + # Nginx 自身健康检查 + location /health { + access_log off; + return 200 'ok'; + add_header Content-Type text/plain; + } + + # 默认代理到 blue,部署时通过 sed 切换 + location / { + proxy_pass http://blue:8080; + proxy_set_header Host $host; + proxy_set_header X-Real-IP $remote_addr; + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + proxy_set_header X-Forwarded-Proto $scheme; + + proxy_connect_timeout 300s; + proxy_send_timeout 300s; + proxy_read_timeout 300s; + + proxy_buffer_size 128k; + proxy_buffers 4 256k; + proxy_busy_buffers_size 256k; + } + } +} diff --git a/client-api/src/main/resources/application-local.yml b/client-api/src/main/resources/application-dev.yml similarity index 100% rename from client-api/src/main/resources/application-local.yml rename to client-api/src/main/resources/application-dev.yml diff --git a/client-api/src/main/resources/application-prod.yml b/client-api/src/main/resources/application-prod.yml new file mode 100644 index 0000000..655a4dc --- /dev/null +++ b/client-api/src/main/resources/application-prod.yml @@ -0,0 +1,118 @@ +# tomcat 端口配置 +server: + port: 8080 +# 数据源配置 +spring: + datasource: + url: jdbc:mysql://${MYSQL_HOST:8.138.5.14}:${MYSQL_PORT:30006}/${DB_NAME:offerpie}?useUnicode=true&characterEncoding=utf-8&allowMultiQueries=true + username: ${MYSQL_USERNAME:root} + password: ${MYSQL_PASSWORD:^CgDatabase2020} + driver-class-name: com.mysql.cj.jdbc.Driver + redis: + host: ${REDIS_HOST:8.138.5.14} + password: ${REDIS_PASSWORD:#8kPCdAsser} + port: ${REDIS_PORT:30089} + timeout: 10s + +# 电子邮箱 +email: + status: close + account: ${EMAIL_ACCOUNT:xxx@163.com} + authorization: ${EMAIL_AUTHORIZATION:123456} + +# 微信支付 +wx_pay: + # status close:关闭 open:打开 + status: close + app_id: your_app_id + merchant_id: ${MERCHANT_ID:your_merchant_id} + private_key: ${WX_PRIVATE_KEY:your_private_key} + merchant_serial_number: ${MERCHANT_SERIAL_NUMBER:your_merchant_serial_number} + api_v3_key: ${API_V3_KEY:your_api_v3_key} + public_key: ${WX_PUBLIC_KEY:your_public_key} + public_key_id: ${PUBLIC_KEY_ID:your_public_key_id} + notify_domain: ${API_NOTIFY_DOMAIN:http://127.0.0.1:8080/api} + +# 支付宝支付 +alipay: + # status close:关闭 open:打开 + status: open + app_id: ${ALIPAY_APP_ID:2021006154620509} + private_key: ${ALIPAY_PRIVATE_KEY:MIIEvAIBADANBgkqhkiG9w0BAQEFAASCBKYwggSiAgEAAoIBAQDWAKDY6yFNVjWH1uXBt+wEqU4QTVAMqwWgMRlniK+kZn4bqL4bwP3sGOB+y4PpJKpFWTiCyl1ChpX311F1tQFhqwJfGedQh2e+OSUaJQmQKJ6JgC/LV8aZPsGSUD/1DwcQUR/fHM3eQ/XFTAb5+o8U7tAkl69FBURP8yuUStUZ3uiz/xFUBbH3RVukrFFbFTany7o1UikqgjY8qj+dWDlqlHr7W1RGVSErtrXnzVVniPb5adol/1Qdw8cy8J1bsiKeRhBjPRDpQX3gb0AR6ThDsP4nZHE02Kw0ApcNGD1/kOSEHgXZiB3KXaIEZnzCf91F3ben8n7Okb49k4bkD6//AgMBAAECggEABUKwsYReJjOHT1aGAVQPFjI/PAFahWUy1zQ9xqDCiI0Ibo+gVLhSxZn68sL39CQ1l0zRKc3lqvv59cSyOO2BIjB892+Akycw8qke81yTtpp5AzJGen4m4J99TKZyXAjJiKol2Wsw2FTdVsM+rKhvGfmg3jVoVo8BleGJYbTRNjiMASdxOzMdlk1j3mKy9OY3AlfC0ERyWloua52bA0LsKTp96iju8uz7sr+1lgQmSC7UJHnVWwKGbsjtcltigCUhuknxqvw7bkBHaEUhuKrOpcKqWW1RRCYhxW1SfKaj2TXIUFaF2rGiJL6wAXkaHp5VZ5kgYksMzCcbofOTjr+XqQKBgQD6h9koWwKWD/ndrMJ3KXsjOQJd8i6bT3QdD5yqOOJjGTS+eAvjpDgsvimRSa9eoyORU0dRUWtgVkTExrZAxoElwZZoiv5XfayKlp56rNLrb6oP58vQfepdexe3KQRjHrRCbKK6YX74Gg+hsL6w6BCagpD2xXBXWk9Ibf0izkLb7QKBgQDarKIdT2cYgAI1YiHkbXuUiTOlCZKw2OeFG0BViGI5c8DOXaxiMe9oeRY6HSBM15s3CmjMZBcjuFe49lhq9Yw4i2Cve8bXiS9Nx0RMVtw/tSY3ztMnjY06Sxj4r0zSTEM0RHp4PxHBCkXvxl6ay5WNxzAnJ5Bw/wA62MVYwEi2GwKBgD/KPUngOn1ZHgcKiQ1sW3Uxe9/N3X4gMfGAWySgjUwsS9i/mk9hLgh+dvkOaP/QzCqhqMQU1iFWdfYgnqFJqfKiIxkFYxOF5Vv4U+vqe14FNakNHUE9/mHFCNnTMNHILJ/JJ3oLfxfPEvHZngRPYoLlAeJDjYJAdXV8w/qLyExVAoGACy2D4+epJ3Fzq3LmtJx8Eq2ovCjvHKcZvo4UDcs46iLmmiXjp+lRiY+W+UV/3y2uO3QGw1viausDYOsGvgDrOAC34oB7O41Bo8Ip1ZUjEQabe4LQ/FaZ/37TOfMMBDvtu4A8mmTu7WUvCgCj2VnKvUKPTZESKxR92IQ4Ij9B+D0CgYBm/JdpxhouDoAt+sNw2pkRu8i7qFQqmiWgE7NLiAOTeIU5n/pGVdkLRV/+B9nEN0vHF8RepToY9kOqWM1jW79yE0whbrexOpJe9gyPQICttN4NlQ46uPnCAtX9Uk07l6UcJ/a4igKFJc4YOkHP3O4Ukb4q7X9ifBuN5m6XkF1JTw==} + alipay_public_key: ${ALIPAY_PUBLIC_KEY:MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAmqQrNof4mHkyHPDbe/CX84xQFobRBGS6L886Z2oWQ2wK4BDMXFna0aH0HhYDb5ZmiMZ+S8UfXwAeiH5XpFSYWCh8iG+Hp0Zjl/NVIBcd4RzCLUI/7Q2jwKFtaYHTcM6yNZxAdX566RDj637ed03FPCGchiQGMZJXJNMdY9L0nyPeNutJ9tszMWMsFUgJsXZt44Gqol5bnpcuPwggnhi0YamRNqGjqQbWABr6AqhsegVV8OsUsQdsfGr22IA/LBmIQcF0YAoTkmBhwiAel1A33xkkcyGARdIgLa+pEGNm59GMbWXOasKg2lETvr9wMfcv1e6yznBfgQ8qMVYx9G6w+QIDAQAB} + notify_domain: ${API_NOTIFY_DOMAIN:http://39.108.134.218:10106/api} + +app: + # 加密秘钥配置 + secret: + token: ${SECRET_TOKEN:Aa123123} + + #登陆配置 + login: + # token配置 + token: + # 过期时间 秒 + exceed_time: 5184000 + #设备在线数量 + device_online_quantity: 3 + + # 短信 + sms: + service_provider: aliyun + aliyun: + access_key_id: LTAI5tJBVUUJhB7yp14UDzVf + access_key_secret: Opf0iO5FKNrdwI63DPhXazW7utAGTj + + oss: + service_provider: aliyun + aliyun: + access_key_id: LTAI5tEdLKKQUKhTyUpfH5Mk + access_key_secret: RjUdTrq0V5qA4b3BUElNhXqs3ZLp5k + + # 防刷配置 20秒 20次 + prevent_replay: + if_open: false + interval_time: 20 + limit_number: 20 + + + + #开放接口 + ignore: + urls: "/public/**,/job/list,/job/detail" + + # 简历配置 + resume: + # 非会员最大限制数量 + max-count: 5 + + # 会员权益奖励配置 + reward: + # 新用户注册赠送会员天数 + register-days: 3 + # 邀请好友奖励会员天数 + invite-days: 7 + + # AI 多供应商配置,第一个为默认 provider + # base-url 配到版本路径,如 DeepSeek: https://api.deepseek.com/v1,豆包: https://ark.cn-beijing.volces.com/api/v3 + ai: + providers: + job-clean: + base-url: ${AI_BASE_URL:https://ark.cn-beijing.volces.com/api/v3} + api-key: ${AI_API_KEY:fd065993-bee2-4f31-8bf2-56d5d3012c02} + model: ${AI_MODEL:doubao-seed-2-0-lite-260215} + job-recommend: + base-url: ${AI_BASE_URL:https://ark.cn-beijing.volces.com/api/v3} + api-key: ${AI_API_KEY:fd065993-bee2-4f31-8bf2-56d5d3012c02} + model: ${AI_MODEL:doubao-1-5-lite-32k-250115} + + + # 岗位清洗配置 + job-clean: + batch-size: 60 + thread-pool-size: 60 + + # 公司数据补充配置 + company-clean: + batch-size: 10 + thread-pool-size: 3 \ No newline at end of file diff --git a/client-api/src/main/resources/application-test.yml b/client-api/src/main/resources/application-test.yml new file mode 100644 index 0000000..b9ae4b2 --- /dev/null +++ b/client-api/src/main/resources/application-test.yml @@ -0,0 +1,118 @@ +# tomcat 端口配置 +server: + port: 8080 +# 数据源配置 +spring: + datasource: + url: jdbc:mysql://${MYSQL_HOST:192.168.31.105}:${MYSQL_PORT:3306}/${DB_NAME:offerpie}?useUnicode=true&characterEncoding=utf-8&allowMultiQueries=true + username: ${MYSQL_USERNAME:root} + password: ${MYSQL_PASSWORD:123456} + driver-class-name: com.mysql.cj.jdbc.Driver + redis: + host: ${REDIS_HOST:192.168.31.105} + password: ${REDIS_PASSWORD:123456} + port: ${REDIS_PORT:6379} + timeout: 8s + +# 电子邮箱 +email: + status: close + account: ${EMAIL_ACCOUNT:xxx@163.com} + authorization: ${EMAIL_AUTHORIZATION:123456} + +# 微信支付 +wx_pay: + # status close:关闭 open:打开 + status: close + app_id: your_app_id + merchant_id: ${MERCHANT_ID:your_merchant_id} + private_key: ${WX_PRIVATE_KEY:your_private_key} + merchant_serial_number: ${MERCHANT_SERIAL_NUMBER:your_merchant_serial_number} + api_v3_key: ${API_V3_KEY:your_api_v3_key} + public_key: ${WX_PUBLIC_KEY:your_public_key} + public_key_id: ${PUBLIC_KEY_ID:your_public_key_id} + notify_domain: ${API_NOTIFY_DOMAIN:http://127.0.0.1:8080/api} + +# 支付宝支付 +alipay: + # status close:关闭 open:打开 + status: open + app_id: ${ALIPAY_APP_ID:2021006154620509} + private_key: ${ALIPAY_PRIVATE_KEY:MIIEvAIBADANBgkqhkiG9w0BAQEFAASCBKYwggSiAgEAAoIBAQDWAKDY6yFNVjWH1uXBt+wEqU4QTVAMqwWgMRlniK+kZn4bqL4bwP3sGOB+y4PpJKpFWTiCyl1ChpX311F1tQFhqwJfGedQh2e+OSUaJQmQKJ6JgC/LV8aZPsGSUD/1DwcQUR/fHM3eQ/XFTAb5+o8U7tAkl69FBURP8yuUStUZ3uiz/xFUBbH3RVukrFFbFTany7o1UikqgjY8qj+dWDlqlHr7W1RGVSErtrXnzVVniPb5adol/1Qdw8cy8J1bsiKeRhBjPRDpQX3gb0AR6ThDsP4nZHE02Kw0ApcNGD1/kOSEHgXZiB3KXaIEZnzCf91F3ben8n7Okb49k4bkD6//AgMBAAECggEABUKwsYReJjOHT1aGAVQPFjI/PAFahWUy1zQ9xqDCiI0Ibo+gVLhSxZn68sL39CQ1l0zRKc3lqvv59cSyOO2BIjB892+Akycw8qke81yTtpp5AzJGen4m4J99TKZyXAjJiKol2Wsw2FTdVsM+rKhvGfmg3jVoVo8BleGJYbTRNjiMASdxOzMdlk1j3mKy9OY3AlfC0ERyWloua52bA0LsKTp96iju8uz7sr+1lgQmSC7UJHnVWwKGbsjtcltigCUhuknxqvw7bkBHaEUhuKrOpcKqWW1RRCYhxW1SfKaj2TXIUFaF2rGiJL6wAXkaHp5VZ5kgYksMzCcbofOTjr+XqQKBgQD6h9koWwKWD/ndrMJ3KXsjOQJd8i6bT3QdD5yqOOJjGTS+eAvjpDgsvimRSa9eoyORU0dRUWtgVkTExrZAxoElwZZoiv5XfayKlp56rNLrb6oP58vQfepdexe3KQRjHrRCbKK6YX74Gg+hsL6w6BCagpD2xXBXWk9Ibf0izkLb7QKBgQDarKIdT2cYgAI1YiHkbXuUiTOlCZKw2OeFG0BViGI5c8DOXaxiMe9oeRY6HSBM15s3CmjMZBcjuFe49lhq9Yw4i2Cve8bXiS9Nx0RMVtw/tSY3ztMnjY06Sxj4r0zSTEM0RHp4PxHBCkXvxl6ay5WNxzAnJ5Bw/wA62MVYwEi2GwKBgD/KPUngOn1ZHgcKiQ1sW3Uxe9/N3X4gMfGAWySgjUwsS9i/mk9hLgh+dvkOaP/QzCqhqMQU1iFWdfYgnqFJqfKiIxkFYxOF5Vv4U+vqe14FNakNHUE9/mHFCNnTMNHILJ/JJ3oLfxfPEvHZngRPYoLlAeJDjYJAdXV8w/qLyExVAoGACy2D4+epJ3Fzq3LmtJx8Eq2ovCjvHKcZvo4UDcs46iLmmiXjp+lRiY+W+UV/3y2uO3QGw1viausDYOsGvgDrOAC34oB7O41Bo8Ip1ZUjEQabe4LQ/FaZ/37TOfMMBDvtu4A8mmTu7WUvCgCj2VnKvUKPTZESKxR92IQ4Ij9B+D0CgYBm/JdpxhouDoAt+sNw2pkRu8i7qFQqmiWgE7NLiAOTeIU5n/pGVdkLRV/+B9nEN0vHF8RepToY9kOqWM1jW79yE0whbrexOpJe9gyPQICttN4NlQ46uPnCAtX9Uk07l6UcJ/a4igKFJc4YOkHP3O4Ukb4q7X9ifBuN5m6XkF1JTw==} + alipay_public_key: ${ALIPAY_PUBLIC_KEY:MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAmqQrNof4mHkyHPDbe/CX84xQFobRBGS6L886Z2oWQ2wK4BDMXFna0aH0HhYDb5ZmiMZ+S8UfXwAeiH5XpFSYWCh8iG+Hp0Zjl/NVIBcd4RzCLUI/7Q2jwKFtaYHTcM6yNZxAdX566RDj637ed03FPCGchiQGMZJXJNMdY9L0nyPeNutJ9tszMWMsFUgJsXZt44Gqol5bnpcuPwggnhi0YamRNqGjqQbWABr6AqhsegVV8OsUsQdsfGr22IA/LBmIQcF0YAoTkmBhwiAel1A33xkkcyGARdIgLa+pEGNm59GMbWXOasKg2lETvr9wMfcv1e6yznBfgQ8qMVYx9G6w+QIDAQAB} + notify_domain: ${API_NOTIFY_DOMAIN:http://39.108.134.218:10106/api} + +app: + # 加密秘钥配置 + secret: + token: ${SECRET_TOKEN:Aa123123} + + #登陆配置 + login: + # token配置 + token: + # 过期时间 秒 + exceed_time: 5184000 + #设备在线数量 + device_online_quantity: 3 + + # 短信 + sms: + service_provider: aliyun + aliyun: + access_key_id: LTAI5tJBVUUJhB7yp14UDzVf + access_key_secret: Opf0iO5FKNrdwI63DPhXazW7utAGTj + + oss: + service_provider: aliyun + aliyun: + access_key_id: LTAI5tEdLKKQUKhTyUpfH5Mk + access_key_secret: RjUdTrq0V5qA4b3BUElNhXqs3ZLp5k + + # 防刷配置 20秒 20次 + prevent_replay: + if_open: false + interval_time: 20 + limit_number: 20 + + + + #开放接口 + ignore: + urls: "/public/**,/job/list,/job/detail" + + # 简历配置 + resume: + # 非会员最大限制数量 + max-count: 5 + + # 会员权益奖励配置 + reward: + # 新用户注册赠送会员天数 + register-days: 3 + # 邀请好友奖励会员天数 + invite-days: 7 + + # AI 多供应商配置,第一个为默认 provider + # base-url 配到版本路径,如 DeepSeek: https://api.deepseek.com/v1,豆包: https://ark.cn-beijing.volces.com/api/v3 + ai: + providers: + job-clean: + base-url: ${AI_BASE_URL:https://ark.cn-beijing.volces.com/api/v3} + api-key: ${AI_API_KEY:fd065993-bee2-4f31-8bf2-56d5d3012c02} + model: ${AI_MODEL:doubao-seed-2-0-lite-260215} + job-recommend: + base-url: ${AI_BASE_URL:https://ark.cn-beijing.volces.com/api/v3} + api-key: ${AI_API_KEY:fd065993-bee2-4f31-8bf2-56d5d3012c02} + model: ${AI_MODEL:doubao-1-5-lite-32k-250115} + + + # 岗位清洗配置 + job-clean: + batch-size: 60 + thread-pool-size: 60 + + # 公司数据补充配置 + company-clean: + batch-size: 10 + thread-pool-size: 3 \ No newline at end of file diff --git a/client-api/src/main/resources/application.yml b/client-api/src/main/resources/application.yml index a36e901..2770430 100644 --- a/client-api/src/main/resources/application.yml +++ b/client-api/src/main/resources/application.yml @@ -12,7 +12,7 @@ spring: application: name: client profiles: - active: ${PROFILES_ACTIVE:local} + active: ${PROFILES_ACTIVE:dev} servlet: multipart: max-file-size: 4MB diff --git a/docker-compose.client-api.yml b/docker-compose.client-api.yml new file mode 100644 index 0000000..231afcd --- /dev/null +++ b/docker-compose.client-api.yml @@ -0,0 +1,82 @@ +services: + nginx: + image: nginx:alpine + container_name: offerpie-backend-client-nginx + restart: unless-stopped + networks: + - xxl-job-network + ports: + - "10202:80" + depends_on: + - blue + - green + healthcheck: + test: ["CMD", "wget", "--spider", "-q", "http://localhost/health"] + interval: 30s + timeout: 10s + retries: 3 + deploy: + resources: + limits: + memory: 512M + cpus: '2' + + blue: + build: + context: . + dockerfile: client-api/Dockerfile + container_name: offerpie-backend-client-blue + restart: unless-stopped + networks: + - xxl-job-network + expose: + - "8080" + environment: + - APP_VERSION=blue + - PROFILES_ACTIVE=prod + - TZ=Asia/Shanghai + healthcheck: + test: ["CMD", "curl", "-f", "http://localhost:8080/api/public/actuator/health"] + interval: 30s + timeout: 10s + retries: 3 + start_period: 40s + volumes: + - /logs/offerpie-backend-client:/app/logs + deploy: + resources: + limits: + memory: 2G + cpus: '2' + + green: + build: + context: . + dockerfile: client-api/Dockerfile + container_name: offerpie-backend-client-green + restart: unless-stopped + networks: + - xxl-job-network + expose: + - "8080" + environment: + - APP_VERSION=green + - PROFILES_ACTIVE=prod + - TZ=Asia/Shanghai + healthcheck: + test: ["CMD", "curl", "-f", "http://localhost:8080/api/public/actuator/health"] + interval: 30s + timeout: 10s + retries: 3 + start_period: 40s + volumes: + - /logs/offerpie-backend-client:/app/logs + deploy: + resources: + limits: + memory: 2G + cpus: '2' + +networks: + xxl-job-network: + external: true diff --git a/settings.xml b/settings.xml new file mode 100644 index 0000000..3dc11f8 --- /dev/null +++ b/settings.xml @@ -0,0 +1,14 @@ + + + + + aliyun + central + aliyun maven + https://maven.aliyun.com/nexus/content/groups/public + + +