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
168 lines
5.2 KiB
PHP
168 lines
5.2 KiB
PHP
<?php
|
||
/**
|
||
* +----------------------------------------------------------------------
|
||
* | CRMEB [ CRMEB赋能开发者,助力企业发展 ]
|
||
* +----------------------------------------------------------------------
|
||
* | Copyright (c) 2016~2022 https://www.crmeb.com All rights reserved.
|
||
* +----------------------------------------------------------------------
|
||
* | Licensed CRMEB并不是自由软件,未经许可不能去掉CRMEB相关版权
|
||
* +----------------------------------------------------------------------
|
||
* | Author: CRMEB Team <admin@crmeb.com>
|
||
* +----------------------------------------------------------------------
|
||
*/
|
||
|
||
namespace crmeb\services\wechat;
|
||
|
||
use Symfony\Contracts\HttpClient\HttpClientInterface;
|
||
use Symfony\Contracts\HttpClient\ResponseInterface;
|
||
use Symfony\Contracts\HttpClient\ResponseStreamInterface;
|
||
use Symfony\Component\HttpClient\NativeHttpClient;
|
||
use Symfony\Component\HttpClient\Response\AsyncContext;
|
||
use Swoole\Coroutine\Http\Client;
|
||
|
||
/**
|
||
* 携程curl请求
|
||
* Class CustomHttpClient
|
||
* @author 等风来
|
||
* @email 136327134@qq.com
|
||
* @date 2024/1/24
|
||
* @package crmeb\services\wechat
|
||
*/
|
||
class CustomHttpClient implements HttpClientInterface
|
||
{
|
||
|
||
/**
|
||
* @var string
|
||
*/
|
||
protected string $baseUrl = '';
|
||
|
||
/**
|
||
* @var int
|
||
*/
|
||
protected int $timeout = 3;
|
||
|
||
/**
|
||
* @var array
|
||
*/
|
||
protected array $httpConfig = [];
|
||
|
||
/**
|
||
* CustomHttpClient constructor.
|
||
* @param string $baseUrl
|
||
* @param array $httpConfig
|
||
*/
|
||
public function __construct(string $baseUrl, array $httpConfig = [])
|
||
{
|
||
$this->baseUrl = $baseUrl;
|
||
$this->httpConfig = $httpConfig;
|
||
}
|
||
|
||
/**
|
||
* 发起请求
|
||
* @param string $method
|
||
* @param string $url
|
||
* @param array $options
|
||
* @return ResponseInterface
|
||
* @author 等风来
|
||
* @email 136327134@qq.com
|
||
* @date 2024/1/24
|
||
*/
|
||
public function request(string $method, string $url, array $options = []): ResponseInterface
|
||
{
|
||
// 合并传入的选项和默认选项
|
||
$options = array_merge(self::OPTIONS_DEFAULTS, $this->httpConfig, $options);
|
||
|
||
$ssl = strstr($this->baseUrl, 'https://') !== false;
|
||
$baseUrl = str_replace(['https://', 'http://', '/'], '', $this->baseUrl);
|
||
|
||
// 使用 Swoole Client 完成请求
|
||
$client = new Client($baseUrl, $ssl ? 443 : 80, $ssl);
|
||
|
||
$client->set([
|
||
'timeout' => !empty($options['timeout']) ? $options['timeout'] : $this->timeout,
|
||
'ssl_cert_file' => $options['cert'] ?? null,
|
||
'ssl_key_file' => $options['ssl_key'] ?? null
|
||
]);
|
||
|
||
$client->setMethod($method);
|
||
$headers = [];
|
||
// 设置请求头
|
||
foreach ($options['headers'] as $key => $value) {
|
||
if (is_string($key)) {
|
||
$values = [];
|
||
if (is_array($value)) {
|
||
foreach ($value as $item) {
|
||
[$type, $val] = strstr($item, ':') !== false ? explode(':', $item) : [null, null];
|
||
if ($type && $val) {
|
||
$values[] = $val;
|
||
}
|
||
}
|
||
$headers[$key] = implode(',', $values);
|
||
} else {
|
||
$values[] = $value;
|
||
$headers[$key] = implode(',', $values);
|
||
}
|
||
} else {
|
||
[$type, $val] = strstr($value, ':') !== false ? explode(':', $value) : [null, null];
|
||
$values[] = $val;
|
||
$headers[$type] = implode(',', $values);
|
||
}
|
||
}
|
||
|
||
$client->setHeaders($headers);
|
||
|
||
if (!empty($options['query'])) {
|
||
$url = $url . (strstr($url, '?') !== false ? '&' : '?') . http_build_query($options['query']);
|
||
}
|
||
if (!empty($options['json'])) {
|
||
$client->setData($options['json']);
|
||
} else if (!empty($options['body'])) {
|
||
$client->setData($options['body']);
|
||
}
|
||
|
||
// 发起请求
|
||
$client->execute('/' . $url);
|
||
|
||
// 创建响应对象
|
||
$response = new SwooleResponse($client);
|
||
|
||
// 关闭客户端连接
|
||
$client->close();
|
||
|
||
return $response;
|
||
}
|
||
|
||
/**
|
||
* @param iterable|ResponseInterface $responses
|
||
* @param float|null $timeout
|
||
* @return ResponseStreamInterface
|
||
* @author 等风来
|
||
* @email 136327134@qq.com
|
||
* @date 2024/1/25
|
||
*/
|
||
public function stream(iterable|ResponseInterface $responses, float $timeout = null): ResponseStreamInterface
|
||
{
|
||
// 创建 ResponseStreamInterface 对象并进行流式处理
|
||
$stream = new AsyncContext($responses, $this->httpClient, $timeout);
|
||
|
||
// 返回流对象
|
||
return $stream;
|
||
}
|
||
|
||
/**
|
||
* @param array $options
|
||
* @return $this
|
||
* @author 等风来
|
||
* @email 136327134@qq.com
|
||
* @date 2024/1/25
|
||
*/
|
||
public function withOptions(array $options): static
|
||
{
|
||
$this->httpConfig = array_merge(self::OPTIONS_DEFAULTS, $options);
|
||
if (!empty($this->httpConfig['base_uri']) && $this->httpConfig['base_uri'] !== $this->baseUrl) {
|
||
$this->baseUrl = $this->httpConfig['base_uri'];
|
||
}
|
||
return $this;
|
||
}
|
||
}
|