Files
huangjingfen/pro_v3.5.1_副本/app/jobs/user/UserBirthdayFormatJob.php
apple 434aa8c69d feat(fsgx): 完成全部24项开发任务 Phase1-7
Phase1 后端核心:
- 新增 fsgx_v1.sql 迁移脚本(is_queue_goods/frozen_points/available_points/no_assess)
- SystemConfigServices 返佣设置扩展(周期人数/分档比例/范围/时机)
- StoreOrderCreateServices 周期循环佣金计算
- StoreOrderTakeServices 佣金发放后同步冻结积分
- StoreProductServices/StoreProduct 保存 is_queue_goods

Phase2 后端接口:
- GET /api/hjf/brokerage/progress 佣金周期进度
- GET /api/hjf/assets/overview 资产总览
- HjfPointsServices 每日 frozen_points 0.4‰ 释放定时任务
- PUT /adminapi/hjf/member/{uid}/no_assess 不考核接口
- GET /adminapi/hjf/points/release_log 积分日志接口

Phase3 前端清理:
- hjfCustom.js 路由精简(仅保留 points/log)
- hjfQueue.js/hjfMember.js API 清理/重定向至 CRMEB 原生接口
- pages.json 公排→推荐佣金/佣金记录/佣金规则

Phase4-5 前端改造:
- queue/status.vue 推荐佣金进度页整体重写
- 商品详情/订单确认/支付结果页文案与逻辑改造
- 个人中心/资产页/引导页/规则页文案改造
- HjfQueueProgress/HjfRefundNotice/HjfAssetCard 组件改造
- 推广中心嵌入佣金进度摘要
- hjfMockData.js 全量更新(公排字段→佣金字段)

Phase6 Admin 增强:
- 用户列表新增 frozen_points/available_points 列及不考核操作按钮
- hjfPoints.js USE_MOCK=false 对接真实积分日志接口

Phase7 配置文档:
- docs/fsgx-phase7-config-checklist.md 后台配置与全链路验收清单

Made-with: Cursor
2026-03-23 22:32:19 +08:00

232 lines
7.3 KiB
PHP
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
<?php
// +----------------------------------------------------------------------
// | CRMEB [ CRMEB赋能开发者助力企业发展 ]
// +----------------------------------------------------------------------
// | Copyright (c) 2016~2026 https://www.crmeb.com All rights reserved.
// +----------------------------------------------------------------------
// | Licensed CRMEB并不是自由软件未经许可不能去掉CRMEB相关版权
// +----------------------------------------------------------------------
// | Author: CRMEB Team <admin@crmeb.com>
// +----------------------------------------------------------------------
namespace app\jobs\user;
use app\services\user\UserServices;
use crmeb\basic\BaseJobs;
use crmeb\traits\QueueTrait;
use think\facade\Log;
/**
* 用户生日格式转换队列任务
* Class UserBirthdayFormatJob
* @package app\jobs\user
*/
class UserBirthdayFormatJob extends BaseJobs
{
use QueueTrait;
/**
* 处理用户生日格式转换
* @param array $userIds 用户ID数组如果为空则处理所有用户
* @param int $batchSize 批处理大小默认100
* @return bool
*/
public function doJob($userIds = [], $batchSize = 100)
{
try {
/** @var UserServices $userServices */
$userServices = app()->make(UserServices::class);
Log::info('开始处理用户生日格式转换任务', [
'user_ids' => $userIds,
'batch_size' => $batchSize
]);
// 如果没有指定用户ID则查询所有需要处理的用户
if (empty($userIds)) {
$userIds = $this->getUsersWithTimestampBirthday($userServices);
}
if (empty($userIds)) {
Log::info('没有找到需要处理的用户生日数据');
return true;
}
// 分批处理用户数据
$chunks = array_chunk($userIds, $batchSize);
$totalChunks = count($chunks);
Log::info('开始分批处理用户生日数据', [
'total_users' => count($userIds),
'total_chunks' => $totalChunks,
'batch_size' => $batchSize
]);
foreach ($chunks as $index => $chunk) {
$this->processBirthdayBatch($userServices, $chunk, $index + 1, $totalChunks);
}
Log::info('用户生日格式转换任务完成', [
'processed_users' => count($userIds)
]);
return true;
} catch (\Throwable $e) {
Log::error('用户生日格式转换任务执行失败', [
'message' => $e->getMessage(),
'file' => $e->getFile(),
'line' => $e->getLine(),
'trace' => $e->getTraceAsString()
]);
return false;
}
}
/**
* 获取生日字段为时间戳格式的用户ID
* @param UserServices $userServices
* @return array
*/
private function getUsersWithTimestampBirthday($userServices)
{
try {
// 查询birthday字段不为空且不为0的用户
// 时间戳通常是10位数字大于9466848002000-01-01的时间戳
$users = $userServices->search([])
->where('birthday', '<>', 0)
->where('is_del', 0)
->column('uid');
Log::info('查询到需要处理的用户', [
'count' => count($users)
]);
return $users;
} catch (\Throwable $e) {
Log::error('查询用户生日数据失败', [
'message' => $e->getMessage(),
'file' => $e->getFile(),
'line' => $e->getLine()
]);
return [];
}
}
/**
* 批量处理用户生日数据
* @param UserServices $userServices
* @param array $userIds
* @param int $currentBatch
* @param int $totalBatches
* @return void
*/
private function processBirthdayBatch($userServices, $userIds, $currentBatch, $totalBatches)
{
try {
Log::info("处理第 {$currentBatch}/{$totalBatches} 批用户数据", [
'user_ids' => $userIds,
'count' => count($userIds)
]);
// 获取用户数据
$users = $userServices->search([])
->whereIn('uid', $userIds)
->where('is_del', 0)
->field('uid,birthday')
->select()
->toArray();
$updateCount = 0;
$errorCount = 0;
foreach ($users as $user) {
try {
$uid = $user['uid'];
$birthday = $user['birthday'];
// 检查是否为时间戳格式10位数字
if (!$this->isTimestamp($birthday)) {
continue;
}
// 将时间戳转换为日期格式 Y-m-d
$birthdayDate = date('Y-m-d', $birthday);
// 更新用户生日数据
$result = $userServices->search([])
->where('uid', $uid)
->update(['birthday' => $birthdayDate]);
if ($result) {
$updateCount++;
Log::debug("用户 {$uid} 生日格式转换成功", [
'old_birthday' => $birthday,
'new_birthday' => $birthdayDate,
'timestamp' => strtotime($birthdayDate)
]);
}
} catch (\Throwable $e) {
$errorCount++;
Log::error("处理用户 {$uid} 生日数据失败", [
'uid' => $uid,
'birthday' => $birthday,
'message' => $e->getMessage()
]);
}
}
Log::info("{$currentBatch} 批处理完成", [
'total_users' => count($users),
'updated_count' => $updateCount,
'error_count' => $errorCount
]);
} catch (\Throwable $e) {
Log::error("批量处理用户生日数据失败", [
'batch' => $currentBatch,
'user_ids' => $userIds,
'message' => $e->getMessage(),
'file' => $e->getFile(),
'line' => $e->getLine()
]);
}
}
/**
* 检查是否为时间戳格式
* @param mixed $value
* @return bool
*/
private function isTimestamp($value)
{
// 检查是否为数字且在合理的时间戳范围内
if (!is_numeric($value)) {
return false;
}
$timestamp = (int)$value;
// 时间戳应该在1970年到2100年之间
return $timestamp > 0 && $timestamp > 946684800 && $timestamp < 4102444800;
}
/**
* 手动触发处理指定用户的生日格式转换
* @param array $userIds 指定的用户ID数组
* @return bool
*/
public function processSpecificUsers($userIds)
{
if (empty($userIds)) {
return false;
}
return $this->doJob($userIds);
}
}