282 lines
8.7 KiB
Go
282 lines
8.7 KiB
Go
|
package COMB
|
|||
|
|
|||
|
import (
|
|||
|
"context"
|
|||
|
"encoding/hex"
|
|||
|
"encoding/json"
|
|||
|
"sync"
|
|||
|
|
|||
|
"tianyuan-api/apps/api/internal/common"
|
|||
|
"tianyuan-api/apps/api/internal/svc"
|
|||
|
"tianyuan-api/apps/api/internal/types"
|
|||
|
"tianyuan-api/apps/api/internal/validator"
|
|||
|
"tianyuan-api/apps/api/internal/westmodel"
|
|||
|
"tianyuan-api/pkg/crypto"
|
|||
|
"tianyuan-api/pkg/errs"
|
|||
|
|
|||
|
"github.com/zeromicro/go-zero/core/logx"
|
|||
|
)
|
|||
|
|
|||
|
type COMB86PMLogic struct {
|
|||
|
logx.Logger
|
|||
|
ctx context.Context
|
|||
|
svcCtx *svc.ServiceContext
|
|||
|
}
|
|||
|
|
|||
|
func NewCOMB86PMLogic(ctx context.Context, svcCtx *svc.ServiceContext) *COMB86PMLogic {
|
|||
|
return &COMB86PMLogic{
|
|||
|
Logger: logx.WithContext(ctx),
|
|||
|
ctx: ctx,
|
|||
|
svcCtx: svcCtx,
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
func (l *COMB86PMLogic) COMB86PM(req *types.Request) (resp string, err *errs.AppError) {
|
|||
|
var status string
|
|||
|
var charges bool
|
|||
|
var remark = ""
|
|||
|
secretKey, ok := l.ctx.Value("secretKey").(string)
|
|||
|
if !ok {
|
|||
|
return "", errs.ErrSystem
|
|||
|
}
|
|||
|
transactionID, ok := l.ctx.Value("transactionID").(string)
|
|||
|
if !ok {
|
|||
|
return "", errs.ErrSystem
|
|||
|
}
|
|||
|
userId, userIdOk := l.ctx.Value("userId").(int64)
|
|||
|
if !userIdOk {
|
|||
|
return "", errs.ErrSystem
|
|||
|
}
|
|||
|
productCode, productCodeOk := l.ctx.Value("productCode").(string)
|
|||
|
if !productCodeOk || productCode == "" {
|
|||
|
return "", errs.ErrSystem
|
|||
|
}
|
|||
|
|
|||
|
defer func() {
|
|||
|
if err != nil {
|
|||
|
status = "failed"
|
|||
|
charges = false
|
|||
|
} else {
|
|||
|
status = "success"
|
|||
|
charges = true
|
|||
|
}
|
|||
|
sendApiRequestMessageErr := l.svcCtx.ApiRequestMqsService.SendApiRequestMessage(l.ctx, transactionID, userId, productCode, status, charges, remark)
|
|||
|
if sendApiRequestMessageErr != nil {
|
|||
|
logx.Errorf("发送 API 请求消息失败: %v", err)
|
|||
|
}
|
|||
|
}()
|
|||
|
// 1、解密
|
|||
|
key, decodeErr := hex.DecodeString(secretKey)
|
|||
|
if decodeErr != nil {
|
|||
|
return "", errs.ErrSystem
|
|||
|
}
|
|||
|
decryptData, aesDecryptErr := crypto.AesDecrypt(req.Data, key)
|
|||
|
if aesDecryptErr != nil || len(decryptData) == 0 {
|
|||
|
return "", errs.ErrParamDecryption
|
|||
|
}
|
|||
|
|
|||
|
// 2、校验
|
|||
|
var data validator.COMB86PMRequest
|
|||
|
if validatorErr := validator.ValidateAndParse(decryptData, &data); validatorErr != nil {
|
|||
|
return "", errs.ErrParamValidation
|
|||
|
}
|
|||
|
|
|||
|
// 准备并发请求
|
|||
|
apiRequests := []APIRequest{
|
|||
|
{SourceId: "G31BJ05", ServiceId: "FLXG9687", Mapping: westmodel.FLXG9687FieldMapping, Wrap: "data", Service: "west"}, // 电诈风险预警
|
|||
|
{SourceId: "RIS031", ServiceId: "FLXG0687", Mapping: westmodel.FLXG0687FieldMapping, Wrap: "", Service: "yushan"}, // 反赌反诈
|
|||
|
{SourceId: "G03HZ01", ServiceId: "FLXG54F5", Mapping: westmodel.FLXG54F5FieldMapping, Wrap: "data", Service: "west"}, // 手机号码风险
|
|||
|
{SourceId: "G22SC01", ServiceId: "FLXG0V4B", Mapping: westmodel.FLXG0V4BFieldMapping, Wrap: "data", Service: "west"}, // 个人涉诉
|
|||
|
{SourceId: "G27BJ05", ServiceId: "JRZQ0A03", Mapping: westmodel.JRZQ0A03FieldMapping, Wrap: "data", Service: "west"}, // 借贷意向
|
|||
|
{SourceId: "G28BJ05", ServiceId: "JRZQ8203", Mapping: westmodel.JRZQ8203FieldMapping, Wrap: "data", Service: "west"}, // 借贷行为
|
|||
|
{SourceId: "G26BJ05", ServiceId: "FLXG3D56", Mapping: westmodel.FLXG3D56FieldMapping, Wrap: "data", Service: "west"}, // 特殊名单
|
|||
|
{SourceId: "G09XM02", ServiceId: "IVYZ5733", Mapping: westmodel.IVYZ5733FieldMapping, Wrap: "data", Service: "west"}, // 单人婚姻
|
|||
|
{SourceId: "G15BJ02", ServiceId: "YYSY6F2E", Mapping: westmodel.YYSY6F2EFieldMapping, Wrap: "data", Service: "west"}, // 运营商三要素详版
|
|||
|
}
|
|||
|
|
|||
|
// 为每个请求构建对应的请求参数
|
|||
|
for i := range apiRequests {
|
|||
|
if apiRequests[i].Service == "west" {
|
|||
|
// 西部服务:先加密后mapping
|
|||
|
westConfig := l.svcCtx.Config.WestConfig
|
|||
|
if apiRequests[i].SourceId == "G05HZ01" {
|
|||
|
// G05HZ01 不需要加密,直接使用原始数据
|
|||
|
dataMap, err := common.StructToMap(data)
|
|||
|
if err != nil {
|
|||
|
logx.Errorf("结构体转map失败:%v", err)
|
|||
|
return "", errs.ErrSystem
|
|||
|
}
|
|||
|
logx.Infof("G05HZ01 原始数据: %+v", data)
|
|||
|
logx.Infof("G05HZ01 转换后数据: %+v", dataMap)
|
|||
|
apiRequests[i].Request = common.MapStructToAPIRequest(dataMap, apiRequests[i].Mapping, apiRequests[i].Wrap)
|
|||
|
logx.Infof("G05HZ01 最终请求数据: %+v", apiRequests[i].Request)
|
|||
|
} else {
|
|||
|
// 其他西部服务需要加密
|
|||
|
encryptedRequest, encryptErr := common.EncryptStructFields(data, westConfig.Key)
|
|||
|
if encryptErr != nil {
|
|||
|
logx.Errorf("西部加密错误:%v", encryptErr)
|
|||
|
return "", errs.ErrSystem
|
|||
|
}
|
|||
|
apiRequests[i].Request = common.MapStructToAPIRequest(encryptedRequest, apiRequests[i].Mapping, apiRequests[i].Wrap)
|
|||
|
}
|
|||
|
} else {
|
|||
|
// 如果是 RIS031,添加 type: 3
|
|||
|
if apiRequests[i].SourceId == "RIS031" {
|
|||
|
apiRequests[i].Request = map[string]interface{}{
|
|||
|
"keyWord": data.IDCard,
|
|||
|
"type": 3,
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
logx.Infof("sourceId:%s,请求参数:%v", apiRequests[i].SourceId, apiRequests[i].Request)
|
|||
|
}
|
|||
|
|
|||
|
// 创建响应通道
|
|||
|
responseChan := make(chan APIResponse, len(apiRequests))
|
|||
|
var wg sync.WaitGroup
|
|||
|
|
|||
|
// 并发处理请求
|
|||
|
for _, apiReq := range apiRequests {
|
|||
|
wg.Add(1)
|
|||
|
go func(req APIRequest) {
|
|||
|
defer wg.Done()
|
|||
|
success := true
|
|||
|
var westResp string
|
|||
|
var callAPIErr *errs.AppError
|
|||
|
|
|||
|
// 根据服务类型选择不同的调用方式
|
|||
|
switch req.Service {
|
|||
|
case "west":
|
|||
|
// 4、发送请求到西部
|
|||
|
logx.Infof("交易号:%s", transactionID)
|
|||
|
var respData []byte
|
|||
|
if req.SourceId == "G05HZ01" {
|
|||
|
respData, callAPIErr = l.svcCtx.WestDexService.CallAPISecond(req.SourceId, req.Request, l.svcCtx.Config.WestConfig.SecretSecondId)
|
|||
|
} else {
|
|||
|
respData, callAPIErr = l.svcCtx.WestDexService.CallAPI(req.SourceId, req.Request, l.svcCtx.Config.WestConfig.SecretId)
|
|||
|
}
|
|||
|
if callAPIErr != nil {
|
|||
|
if callAPIErr.Code == errs.ErrDataSource.Code {
|
|||
|
// 数据源错误(如查询无结果)是业务正常情况,记录为 info
|
|||
|
logx.Infof("西部请求业务状态:sourceId:%s, resp:%v", req.SourceId, respData)
|
|||
|
var jsonCheck interface{}
|
|||
|
if json.Unmarshal(respData, &jsonCheck) != nil {
|
|||
|
westResp = "{}"
|
|||
|
} else {
|
|||
|
westResp = string(respData)
|
|||
|
}
|
|||
|
} else {
|
|||
|
// 其他业务错误
|
|||
|
logx.Errorf("西部请求业务错误:sourceId:%s,err:%v, resp:%v", req.SourceId, callAPIErr, respData)
|
|||
|
westResp = "{}"
|
|||
|
}
|
|||
|
} else {
|
|||
|
// 确保返回的是有效的 JSON
|
|||
|
if len(respData) == 0 {
|
|||
|
westResp = "{}"
|
|||
|
} else {
|
|||
|
// 验证 JSON 是否有效
|
|||
|
var jsonCheck interface{}
|
|||
|
if json.Unmarshal(respData, &jsonCheck) != nil {
|
|||
|
westResp = "{}"
|
|||
|
} else {
|
|||
|
westResp = string(respData)
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
case "yushan":
|
|||
|
respData, err := l.svcCtx.YushanService.Request(req.SourceId, req.Request)
|
|||
|
if err != nil {
|
|||
|
logx.Errorf("羽山请求失败:sourceId:%s,err:%v", req.SourceId, err)
|
|||
|
if appErr, ok := err.(*errs.AppError); ok {
|
|||
|
callAPIErr = appErr
|
|||
|
} else {
|
|||
|
callAPIErr = errs.ErrSystem
|
|||
|
}
|
|||
|
westResp = "{}" // 发生错误时返回空对象
|
|||
|
} else {
|
|||
|
// 确保返回的是有效的 JSON
|
|||
|
if len(respData) == 0 {
|
|||
|
westResp = "{}"
|
|||
|
} else {
|
|||
|
// 验证 JSON 是否有效
|
|||
|
var jsonCheck interface{}
|
|||
|
if json.Unmarshal(respData, &jsonCheck) != nil {
|
|||
|
westResp = "{}"
|
|||
|
} else {
|
|||
|
westResp = string(respData)
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
default:
|
|||
|
success = false
|
|||
|
westResp = "{}"
|
|||
|
logx.Errorf("未知的服务类型:%s", req.Service)
|
|||
|
}
|
|||
|
|
|||
|
if callAPIErr != nil {
|
|||
|
success = false
|
|||
|
}
|
|||
|
|
|||
|
// 确保响应是有效的 JSON
|
|||
|
var jsonResp json.RawMessage
|
|||
|
if err := json.Unmarshal([]byte(westResp), &jsonResp); err != nil {
|
|||
|
jsonResp = json.RawMessage("{}")
|
|||
|
}
|
|||
|
|
|||
|
responseChan <- APIResponse{
|
|||
|
ServiceId: req.ServiceId,
|
|||
|
Resp: jsonResp,
|
|||
|
Success: success,
|
|||
|
}
|
|||
|
}(apiReq)
|
|||
|
}
|
|||
|
|
|||
|
// 等待所有请求完成
|
|||
|
go func() {
|
|||
|
wg.Wait()
|
|||
|
close(responseChan)
|
|||
|
}()
|
|||
|
|
|||
|
// 处理响应
|
|||
|
var responses []APIResponse
|
|||
|
for resp := range responseChan {
|
|||
|
responses = append(responses, resp)
|
|||
|
}
|
|||
|
|
|||
|
responseData := ResponseData{
|
|||
|
Responses: make([]struct {
|
|||
|
ServiceId string `json:"api_code"`
|
|||
|
Data json.RawMessage `json:"data"`
|
|||
|
Success bool `json:"success"`
|
|||
|
}, len(responses)),
|
|||
|
}
|
|||
|
|
|||
|
for i, resp := range responses {
|
|||
|
responseData.Responses[i] = struct {
|
|||
|
ServiceId string `json:"api_code"`
|
|||
|
Data json.RawMessage `json:"data"`
|
|||
|
Success bool `json:"success"`
|
|||
|
}{
|
|||
|
ServiceId: resp.ServiceId,
|
|||
|
Data: resp.Resp,
|
|||
|
Success: resp.Success,
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// 将响应数据转换为JSON
|
|||
|
jsonData, marshalErr := json.Marshal(responseData)
|
|||
|
if marshalErr != nil {
|
|||
|
logx.Errorf("JSON编码错误:%v", marshalErr)
|
|||
|
return "", errs.ErrSystem
|
|||
|
}
|
|||
|
|
|||
|
// 加密JSON数据
|
|||
|
encryptData, aesEncrypt := crypto.AesEncrypt(jsonData, key)
|
|||
|
if aesEncrypt != nil {
|
|||
|
return "", errs.ErrSystem
|
|||
|
}
|
|||
|
|
|||
|
return string(encryptData), nil
|
|||
|
}
|