This commit is contained in:
Mrx
2026-05-23 16:13:30 +08:00
parent effe82ce65
commit e2aab6af71
11 changed files with 695 additions and 9 deletions

View File

@@ -22,8 +22,31 @@ service main {
@handler getAppConfig
get /app/config returns (getAppConfigResp)
@handler getHomeDynamicData
get /app/home/dynamic (getHomeDynamicDataReq) returns (getHomeDynamicDataResp)
}
type (
getHomeDynamicDataReq {
LastId int64 `json:"lastId,optional"`
}
InquiryRecordItem {
Id int64 `json:"id"`
Tag string `json:"tag"`
Vin string `json:"vin"`
Model string `json:"model"`
}
ReviewItem {
Name string `json:"name"`
Content string `json:"content"`
}
getHomeDynamicDataResp {
Cases []InquiryRecordItem `json:"cases"`
Reviews []ReviewItem `json:"reviews"`
}
)
type (
// 心跳检测响应
HealthCheckResp {

View File

@@ -0,0 +1,29 @@
package app
import (
"net/http"
"qnc-server/app/main/api/internal/logic/app"
"qnc-server/app/main/api/internal/svc"
"qnc-server/app/main/api/internal/types"
"github.com/zeromicro/go-zero/rest/httpx"
)
func GetHomeDynamicDataHandler(svcCtx *svc.ServiceContext) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
var req types.GetHomeDynamicDataReq
if err := httpx.Parse(r, &req); err != nil {
httpx.ErrorCtx(r.Context(), w, err)
return
}
l := app.NewGetHomeDynamicDataLogic(r.Context(), svcCtx)
resp, err := l.GetHomeDynamicData(&req)
if err != nil {
httpx.ErrorCtx(r.Context(), w, err)
} else {
httpx.OkJsonCtx(r.Context(), w, resp)
}
}
}

View File

@@ -842,6 +842,11 @@ func RegisterHandlers(server *rest.Server, serverCtx *svc.ServiceContext) {
Path: "/app/version",
Handler: app.GetAppVersionHandler(serverCtx),
},
{
Method: http.MethodGet,
Path: "/app/home/dynamic",
Handler: app.GetHomeDynamicDataHandler(serverCtx),
},
{
// 心跳检测接口
Method: http.MethodGet,

View File

@@ -0,0 +1,64 @@
package app
import (
"context"
"qnc-server/app/main/api/internal/svc"
"qnc-server/app/main/api/internal/types"
"qnc-server/app/main/model"
"github.com/zeromicro/go-zero/core/logx"
)
type GetHomeDynamicDataLogic struct {
logx.Logger
ctx context.Context
svcCtx *svc.ServiceContext
}
func NewGetHomeDynamicDataLogic(ctx context.Context, svcCtx *svc.ServiceContext) *GetHomeDynamicDataLogic {
return &GetHomeDynamicDataLogic{
Logger: logx.WithContext(ctx),
ctx: ctx,
svcCtx: svcCtx,
}
}
func (l *GetHomeDynamicDataLogic) GetHomeDynamicData(req *types.GetHomeDynamicDataReq) (resp *types.GetHomeDynamicDataResp, err error) {
// 1. 获取真实查询案例 (支持增量查询)
builder := l.svcCtx.InquiryRecordModel.SelectBuilder().Where("status = ?", 1)
var records []*model.InquiryRecord
if req.LastId > 0 {
// 增量获取大于 lastId 的记录
records, err = l.svcCtx.InquiryRecordModel.FindPageListByIdASC(l.ctx, builder, req.LastId, 50)
} else {
// 首次加载获取最新的 10 条
records, err = l.svcCtx.InquiryRecordModel.FindPageListByPage(l.ctx, builder, 1, 10, "id DESC")
}
cases := make([]types.InquiryRecordItem, 0)
if err == nil {
for _, r := range records {
cases = append(cases, types.InquiryRecordItem{
Id: r.Id,
Tag: r.InquiryTag,
Vin: r.VinMasked,
Model: r.CarModel,
})
}
}
// 2. 硬编码返回一些初始评价
reviews := []types.ReviewItem{
{Name: "王先生", Content: "查询速度非常快,报告里关于出险的描述非常详细,避坑神器!"},
{Name: "李女士", Content: "买二手车之前查一下真的有必要,帮我发现了一台调表车。"},
{Name: "张先生", Content: "数据更新很及时4S店的维保记录全都能查到。"},
{Name: "赵先生", Content: "操作简单,手机号绑定后报告一直保存着,随时可以查看。"},
}
return &types.GetHomeDynamicDataResp{
Cases: cases,
Reviews: reviews,
}, nil
}

View File

@@ -5,10 +5,12 @@ import (
"qnc-server/app/main/api/internal/svc"
"qnc-server/app/main/api/internal/types"
"qnc-server/app/main/model"
"qnc-server/common/xerr"
"github.com/pkg/errors"
"github.com/zeromicro/go-zero/core/logx"
"github.com/zeromicro/go-zero/core/threading"
)
type ToolboxQueryLogic struct {
@@ -40,8 +42,54 @@ func (l *ToolboxQueryLogic) ToolboxQuery(req *types.ToolboxQueryReq) (*types.Too
if err != nil {
return nil, err
}
// 记录免费工具查询到动态展示表
l.recordFreeToolInquiry(req.ToolKey)
return &types.ToolboxQueryResp{
ToolKey: req.ToolKey,
Result: result,
}, nil
}
// recordFreeToolInquiry 记录免费工具查询
func (l *ToolboxQueryLogic) recordFreeToolInquiry(toolKey string) {
// 异步记录,不影响主流程
threading.GoSafe(func() {
ctx := context.Background()
// 1. 获取工具名称 (这里可以建立一个简单的映射,或者从前端传过来,或者查询配置)
// 为了简单和解耦,我们这里仅记录 ToolKey前端展示时再映射名称或者在这里硬编码一些常用名称
toolName := toolKey
// 如果能访问到 toolboxRegistry 的逻辑最好,但后端通常不包含前端配置
// 我们可以通过一个简单的 Switch 处理一些核心工具名
switch toolKey {
case "ip-location":
toolName = "IP地址查询"
case "idcard-info":
toolName = "身份证归属地"
case "oilprice":
toolName = "实时油价"
case "joke":
toolName = "趣味笑话"
case "dream":
toolName = "周公解梦"
case "constellation":
toolName = "星座运势"
case "garbage":
toolName = "垃圾分类"
default:
// 其他的就直接用 key 或保持原样
}
record := &model.InquiryRecord{
UserPhoneTail: "系统",
DisplayName: "游客用户",
VinMasked: "******",
CarModel: "免费工具查询",
InquiryTag: toolName,
Status: 1,
}
_, _ = l.svcCtx.InquiryRecordModel.Insert(ctx, nil, record)
})
}

View File

@@ -6,13 +6,13 @@ import (
"encoding/json"
"fmt"
"os"
"regexp"
"strings"
"qnc-server/app/main/api/internal/svc"
"qnc-server/app/main/api/internal/types"
"qnc-server/app/main/model"
"qnc-server/pkg/lzkit/crypto"
"qnc-server/pkg/lzkit/lzUtils"
"regexp"
"strings"
"github.com/google/uuid"
"github.com/hibiken/asynq"
@@ -186,6 +186,9 @@ func (l *PaySuccessNotifyUserHandler) ProcessTask(ctx context.Context, t *asynq.
return l.handleError(ctx, updateQueryErr, order, query)
}
// 记录动态查询记录 (真实案例展示)
go l.recordInquiryRecord(context.Background(), decryptData, product, encryptData)
// 报告生成成功后,发送代理处理异步任务(不阻塞报告流程)
if asyncErr := l.svcCtx.AsynqService.SendAgentProcessTask(order.Id); asyncErr != nil {
// 代理处理任务发送失败,只记录日志,不影响报告流程
@@ -377,6 +380,84 @@ func maskPhone(phone string) string {
return phone[:3] + strings.Repeat("*", length-7) + phone[length-4:]
}
// recordInquiryRecord 记录成功的查询到动态展示表
func (l *PaySuccessNotifyUserHandler) recordInquiryRecord(ctx context.Context, decryptParams []byte, product *model.Product, encryptedResp string) {
defer func() {
if r := recover(); r != nil {
logx.Errorf("记录查询记录异常: %v", r)
}
}()
secretKey := l.svcCtx.Config.Encrypt.SecretKey
key, _ := hex.DecodeString(secretKey)
// 1. 解析参数获取手机号和VIN
var params map[string]interface{}
_ = json.Unmarshal(decryptParams, &params)
mobile, _ := params["mobile"].(string)
vin, _ := params["vin_code"].(string)
if vin == "" {
vin, _ = params["vin"].(string)
}
if mobile == "" || vin == "" {
return
}
// 2. 解密响应数据获取车型
decryptResp, err := crypto.AesDecrypt(encryptedResp, key)
if err != nil {
return
}
// 尝试从响应中寻找车型信息 (这里根据实际响应结构寻找)
// 通常在车辆信息接口的 data 字段中
carModel := "未知车型"
respStr := string(decryptResp)
// 简单的规则寻找车型字段
if strings.Contains(respStr, "model_name") {
re := regexp.MustCompile(`"model_name"\s*:\s*"([^"]+)"`)
match := re.FindStringSubmatch(respStr)
if len(match) > 1 {
carModel = match[1]
}
} else if strings.Contains(respStr, "brand_name") {
re := regexp.MustCompile(`"brand_name"\s*:\s*"([^"]+)"`)
match := re.FindStringSubmatch(respStr)
if len(match) > 1 {
carModel = match[1]
}
}
// 3. 脱敏处理
phoneTail := ""
if len(mobile) >= 4 {
phoneTail = mobile[len(mobile)-4:]
}
displayName := "用户*" + phoneTail
maskedVin := ""
if len(vin) >= 8 {
maskedVin = vin[:4] + "********" + vin[len(vin)-4:]
} else {
maskedVin = vin
}
// 4. 保存记录
record := &model.InquiryRecord{
UserPhoneTail: phoneTail,
DisplayName: displayName,
VinMasked: maskedVin,
CarModel: carModel,
InquiryTag: product.ProductName,
Status: 1,
}
_, _ = l.svcCtx.InquiryRecordModel.Insert(ctx, nil, record)
}
// 通用敏感信息脱敏 - 根据字符串长度比例进行脱敏
func maskGeneral(value string) string {
length := len(value)

View File

@@ -82,6 +82,7 @@ type ServiceContext struct {
ExampleModel model.ExampleModel
GlobalNotificationsModel model.GlobalNotificationsModel
AuthorizationDocumentModel model.AuthorizationDocumentModel
InquiryRecordModel model.InquiryRecordModel
// 第三方服务
TianyuanapiCallLogService *service.TianyuanapiCallLogService
@@ -172,6 +173,7 @@ func NewServiceContext(c config.Config) *ServiceContext {
exampleModel := model.NewExampleModel(db, cacheConf)
globalNotificationsModel := model.NewGlobalNotificationsModel(db, cacheConf)
authorizationDocumentModel := model.NewAuthorizationDocumentModel(db, cacheConf)
inquiryRecordModel := model.NewInquiryRecordModel(db, cacheConf)
tianyuanapiCallLogModel := model.NewTianyuanapiCallLogModel(db, cacheConf)
// ============================== 第三方服务初始化 ==============================
@@ -297,6 +299,7 @@ func NewServiceContext(c config.Config) *ServiceContext {
ExampleModel: exampleModel,
GlobalNotificationsModel: globalNotificationsModel,
AuthorizationDocumentModel: authorizationDocumentModel,
InquiryRecordModel: inquiryRecordModel,
// 第三方服务
TianyuanapiCallLogService: tianyuanapiCallLogService,

View File

@@ -2436,3 +2436,24 @@ type SendSmsReq struct {
ActionType string `json:"actionType" validate:"required,oneof=login register query agentApply realName bindMobile"`
CaptchaVerifyParam string `json:"captchaVerifyParam,optional"`
}
type GetHomeDynamicDataReq struct {
LastId int64 `json:"lastId,optional"`
}
type InquiryRecordItem struct {
Id int64 `json:"id"`
Tag string `json:"tag"`
Vin string `json:"vin"`
Model string `json:"model"`
}
type ReviewItem struct {
Name string `json:"name"`
Content string `json:"content"`
}
type GetHomeDynamicDataResp struct {
Cases []InquiryRecordItem `json:"cases"`
Reviews []ReviewItem `json:"reviews"`
}