2025-01-10 00:09:25 +08:00
|
|
|
|
package query
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"context"
|
|
|
|
|
"encoding/hex"
|
2025-04-11 13:10:17 +08:00
|
|
|
|
"qnc-server/app/user/cmd/api/internal/svc"
|
|
|
|
|
"qnc-server/app/user/cmd/api/internal/types"
|
|
|
|
|
"qnc-server/common/xerr"
|
2025-04-22 02:11:56 +08:00
|
|
|
|
"qnc-server/pkg/lzkit/crypto"
|
2025-04-11 13:10:17 +08:00
|
|
|
|
|
2025-04-22 02:11:56 +08:00
|
|
|
|
"github.com/bytedance/sonic"
|
2025-01-10 00:09:25 +08:00
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
|
|
|
|
|
|
"github.com/zeromicro/go-zero/core/logx"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
type QueryExampleLogic struct {
|
|
|
|
|
logx.Logger
|
|
|
|
|
ctx context.Context
|
|
|
|
|
svcCtx *svc.ServiceContext
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func NewQueryExampleLogic(ctx context.Context, svcCtx *svc.ServiceContext) *QueryExampleLogic {
|
|
|
|
|
return &QueryExampleLogic{
|
|
|
|
|
Logger: logx.WithContext(ctx),
|
|
|
|
|
ctx: ctx,
|
|
|
|
|
svcCtx: svcCtx,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (l *QueryExampleLogic) QueryExample(req *types.QueryExampleReq) (resp *types.QueryExampleResp, err error) {
|
2025-04-22 02:11:56 +08:00
|
|
|
|
// 根据产品特性标识获取产品信息
|
|
|
|
|
product, err := l.svcCtx.ProductModel.FindOneByProductEn(l.ctx, req.Feature)
|
2025-01-10 00:09:25 +08:00
|
|
|
|
if err != nil {
|
2025-04-22 02:11:56 +08:00
|
|
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.SERVER_COMMON_ERROR), "示例报告, 获取商品信息失败, %v", err)
|
2025-01-10 00:09:25 +08:00
|
|
|
|
}
|
2025-04-22 02:11:56 +08:00
|
|
|
|
|
|
|
|
|
// 创建一个空的Query结构体来存储结果
|
|
|
|
|
query := types.Query{
|
|
|
|
|
ProductName: product.ProductName,
|
|
|
|
|
QueryData: make([]types.QueryItem, 0),
|
|
|
|
|
QueryParams: make(map[string]interface{}),
|
2025-01-10 00:09:25 +08:00
|
|
|
|
}
|
2025-04-22 02:11:56 +08:00
|
|
|
|
query.QueryParams = map[string]interface{}{
|
|
|
|
|
"id_card": "45000000000000000",
|
|
|
|
|
"mobile": "13700000000",
|
|
|
|
|
"name": "张老三",
|
2025-01-26 14:28:36 +08:00
|
|
|
|
}
|
2025-04-22 02:11:56 +08:00
|
|
|
|
// 查询ProductFeatureModel获取产品相关的功能列表
|
|
|
|
|
builder := l.svcCtx.ProductFeatureModel.SelectBuilder().Where("product_id = ?", product.Id)
|
|
|
|
|
productFeatures, err := l.svcCtx.ProductFeatureModel.FindAll(l.ctx, builder, "")
|
2025-01-10 00:09:25 +08:00
|
|
|
|
if err != nil {
|
2025-04-22 02:11:56 +08:00
|
|
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.SERVER_COMMON_ERROR), "示例报告, 查询 ProductFeatureModel 错误: %v", err)
|
2025-01-10 00:09:25 +08:00
|
|
|
|
}
|
2025-04-22 02:11:56 +08:00
|
|
|
|
// 从每个启用的特性获取示例数据并合并
|
|
|
|
|
for _, pf := range productFeatures {
|
|
|
|
|
if pf.Enable != 1 {
|
|
|
|
|
continue // 跳过未启用的特性
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 根据特性ID查找示例数据
|
|
|
|
|
example, err := l.svcCtx.ExampleModel.FindOneByFeatureId(l.ctx, pf.FeatureId)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logx.Infof("示例报告, 特性ID %d 无示例数据: %v", pf.FeatureId, err)
|
|
|
|
|
continue // 如果没有示例数据就跳过
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 获取对应的Feature信息
|
|
|
|
|
feature, err := l.svcCtx.FeatureModel.FindOne(l.ctx, pf.FeatureId)
|
|
|
|
|
if err != nil {
|
|
|
|
|
logx.Infof("示例报告, 无法获取特性ID %d 的信息: %v", pf.FeatureId, err)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var queryItem types.QueryItem
|
|
|
|
|
|
|
|
|
|
// 解密查询数据
|
|
|
|
|
secretKey := l.svcCtx.Config.Encrypt.SecretKey
|
|
|
|
|
key, decodeErr := hex.DecodeString(secretKey)
|
|
|
|
|
if decodeErr != nil {
|
|
|
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.SERVER_COMMON_ERROR), "示例报告, 获取AES解密解药失败, %v", err)
|
|
|
|
|
}
|
|
|
|
|
// 解析示例内容
|
|
|
|
|
if example.Content == "000" {
|
|
|
|
|
queryItem.Data = example.Content
|
|
|
|
|
} else {
|
|
|
|
|
// 解密数据
|
|
|
|
|
decryptedData, decryptErr := crypto.AesDecrypt(example.Content, key)
|
|
|
|
|
if decryptErr != nil {
|
|
|
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.SERVER_COMMON_ERROR), "示例报告, 解密数据失败: %v", decryptErr)
|
|
|
|
|
}
|
|
|
|
|
err = sonic.Unmarshal([]byte(decryptedData), &queryItem.Data)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, errors.Wrapf(xerr.NewErrCode(xerr.SERVER_COMMON_ERROR), "示例报告, 解析示例内容失败: %v", err)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 添加特性信息
|
|
|
|
|
queryItem.Feature = map[string]interface{}{
|
|
|
|
|
"featureName": feature.Name,
|
|
|
|
|
"sort": pf.Sort,
|
|
|
|
|
}
|
|
|
|
|
// 添加到查询数据中
|
|
|
|
|
query.QueryData = append(query.QueryData, queryItem)
|
2025-01-10 00:09:25 +08:00
|
|
|
|
}
|
2025-04-22 02:11:56 +08:00
|
|
|
|
|
2025-01-10 00:09:25 +08:00
|
|
|
|
return &types.QueryExampleResp{
|
|
|
|
|
Query: query,
|
|
|
|
|
}, nil
|
|
|
|
|
}
|
2025-04-22 02:11:56 +08:00
|
|
|
|
|
|
|
|
|
// 如果有重复声明的问题,可能需要将这个函数移动到公共位置或者确认是否已存在
|
|
|
|
|
// 暂时保留,如果仍有问题可以考虑将其移到其他文件
|
|
|
|
|
/* func (l *QueryExampleLogic) UpdateFeatureAndProductFeature(productID int64, target *[]types.QueryItem) error {
|
2025-01-26 14:28:36 +08:00
|
|
|
|
// 遍历 target 数组,使用倒序遍历,以便删除元素时不影响索引
|
|
|
|
|
for i := len(*target) - 1; i >= 0; i-- {
|
|
|
|
|
queryItem := &(*target)[i]
|
|
|
|
|
|
|
|
|
|
// 确保 Data 为 map 类型
|
|
|
|
|
data, ok := queryItem.Data.(map[string]interface{})
|
|
|
|
|
if !ok {
|
|
|
|
|
return fmt.Errorf("queryItem.Data 必须是 map[string]interface{} 类型")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 从 Data 中获取 apiID
|
|
|
|
|
apiID, ok := data["apiID"].(string)
|
|
|
|
|
if !ok {
|
|
|
|
|
return fmt.Errorf("queryItem.Data 中的 apiID 必须是字符串类型")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 查询 Feature
|
|
|
|
|
feature, err := l.svcCtx.FeatureModel.FindOneByApiId(l.ctx, apiID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
// 如果 Feature 查不到,也要删除当前 QueryItem
|
|
|
|
|
*target = append((*target)[:i], (*target)[i+1:]...)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 查询 ProductFeatureModel
|
|
|
|
|
builder := l.svcCtx.ProductFeatureModel.SelectBuilder().Where("product_id = ?", productID)
|
|
|
|
|
productFeatures, err := l.svcCtx.ProductFeatureModel.FindAll(l.ctx, builder, "")
|
|
|
|
|
if err != nil {
|
|
|
|
|
return fmt.Errorf("查询 ProductFeatureModel 错误: %v", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 遍历 productFeatures,找到与 feature.ID 关联且 enable == 1 的项
|
|
|
|
|
var featureData map[string]interface{}
|
|
|
|
|
foundFeature := false
|
|
|
|
|
|
|
|
|
|
for _, pf := range productFeatures {
|
|
|
|
|
if pf.FeatureId == feature.Id { // 确保和 Feature 关联
|
|
|
|
|
foundFeature = true
|
|
|
|
|
if pf.Enable == 1 {
|
|
|
|
|
featureData = map[string]interface{}{
|
|
|
|
|
"featureName": feature.Name,
|
|
|
|
|
"sort": pf.Sort,
|
|
|
|
|
}
|
|
|
|
|
break // 找到第一个符合条件的就退出循环
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 如果没有符合条件的 feature 或者 featureData 为空,则删除当前 queryItem
|
|
|
|
|
if !foundFeature || featureData == nil {
|
|
|
|
|
*target = append((*target)[:i], (*target)[i+1:]...)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 更新 queryItem 的 Feature 字段(不是数组)
|
|
|
|
|
queryItem.Feature = featureData
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return nil
|
2025-04-22 02:11:56 +08:00
|
|
|
|
} */
|