2025-07-28 01:46:39 +08:00
|
|
|
|
package yushan
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
|
"bytes"
|
2025-08-25 15:44:06 +08:00
|
|
|
|
"context"
|
2025-07-28 01:46:39 +08:00
|
|
|
|
"crypto/aes"
|
|
|
|
|
|
"crypto/cipher"
|
2025-08-25 15:44:06 +08:00
|
|
|
|
"crypto/md5"
|
2025-07-28 01:46:39 +08:00
|
|
|
|
"crypto/rand"
|
|
|
|
|
|
"encoding/base64"
|
|
|
|
|
|
"encoding/hex"
|
|
|
|
|
|
"encoding/json"
|
|
|
|
|
|
"errors"
|
|
|
|
|
|
"fmt"
|
|
|
|
|
|
"io"
|
|
|
|
|
|
"net/http"
|
|
|
|
|
|
"strings"
|
|
|
|
|
|
"time"
|
|
|
|
|
|
|
2025-08-25 15:44:06 +08:00
|
|
|
|
"tyapi-server/internal/shared/external_logger"
|
|
|
|
|
|
|
2025-07-28 01:46:39 +08:00
|
|
|
|
"github.com/tidwall/gjson"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
var (
|
|
|
|
|
|
ErrDatasource = errors.New("数据源异常")
|
|
|
|
|
|
ErrNotFound = errors.New("查询为空")
|
|
|
|
|
|
ErrSystem = errors.New("系统异常")
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
type YushanConfig struct {
|
|
|
|
|
|
URL string
|
|
|
|
|
|
ApiKey string
|
|
|
|
|
|
AcctID string
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
type YushanService struct {
|
|
|
|
|
|
config YushanConfig
|
2025-08-25 15:44:06 +08:00
|
|
|
|
logger *external_logger.ExternalServiceLogger
|
2025-07-28 01:46:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-08-25 15:44:06 +08:00
|
|
|
|
// NewYushanService 是一个构造函数,用于初始化 YushanService
|
|
|
|
|
|
func NewYushanService(url, apiKey, acctID string, logger *external_logger.ExternalServiceLogger) *YushanService {
|
2025-07-28 01:46:39 +08:00
|
|
|
|
return &YushanService{
|
|
|
|
|
|
config: YushanConfig{
|
|
|
|
|
|
URL: url,
|
|
|
|
|
|
ApiKey: apiKey,
|
|
|
|
|
|
AcctID: acctID,
|
|
|
|
|
|
},
|
2025-08-25 15:44:06 +08:00
|
|
|
|
logger: logger,
|
2025-07-28 01:46:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-08-25 15:44:06 +08:00
|
|
|
|
// CallAPI 调用羽山数据的 API
|
|
|
|
|
|
func (y *YushanService) CallAPI(ctx context.Context, code string, params map[string]interface{}) (respBytes []byte, err error) {
|
|
|
|
|
|
startTime := time.Now()
|
|
|
|
|
|
requestID := y.generateRequestID()
|
|
|
|
|
|
|
|
|
|
|
|
// 从ctx中获取transactionId
|
|
|
|
|
|
var transactionID string
|
|
|
|
|
|
if ctxTransactionID, ok := ctx.Value("transaction_id").(string); ok {
|
|
|
|
|
|
transactionID = ctxTransactionID
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 记录请求日志
|
|
|
|
|
|
if y.logger != nil {
|
|
|
|
|
|
y.logger.LogRequest(requestID, code, y.config.URL, y.buildLogData(params, transactionID))
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-28 01:46:39 +08:00
|
|
|
|
// 获取当前时间戳
|
|
|
|
|
|
unixMilliseconds := time.Now().UnixNano() / int64(time.Millisecond)
|
|
|
|
|
|
|
|
|
|
|
|
// 生成请求序列号
|
|
|
|
|
|
requestSN, _ := y.GenerateRandomString()
|
|
|
|
|
|
|
|
|
|
|
|
// 构建请求数据
|
|
|
|
|
|
reqData := map[string]interface{}{
|
|
|
|
|
|
"prod_id": code,
|
|
|
|
|
|
"req_time": unixMilliseconds,
|
|
|
|
|
|
"request_sn": requestSN,
|
|
|
|
|
|
"req_data": params,
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 将请求数据转换为 JSON 字节数组
|
|
|
|
|
|
messageBytes, err := json.Marshal(reqData)
|
|
|
|
|
|
if err != nil {
|
2025-08-25 15:44:06 +08:00
|
|
|
|
err = fmt.Errorf("%w: %s", ErrSystem, err.Error())
|
|
|
|
|
|
if y.logger != nil {
|
|
|
|
|
|
y.logger.LogError(requestID, code, err, y.buildLogData(params, transactionID))
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil, err
|
2025-07-28 01:46:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 获取 API 密钥
|
|
|
|
|
|
key, err := hex.DecodeString(y.config.ApiKey)
|
|
|
|
|
|
if err != nil {
|
2025-08-25 15:44:06 +08:00
|
|
|
|
err = fmt.Errorf("%w: %s", ErrSystem, err.Error())
|
|
|
|
|
|
if y.logger != nil {
|
|
|
|
|
|
y.logger.LogError(requestID, code, err, y.buildLogData(params, transactionID))
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil, err
|
2025-07-28 01:46:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 使用 AES CBC 加密请求数据
|
|
|
|
|
|
cipherText := y.AES_CBC_Encrypt(messageBytes, key)
|
|
|
|
|
|
|
|
|
|
|
|
// 将加密后的数据编码为 Base64 字符串
|
|
|
|
|
|
content := base64.StdEncoding.EncodeToString(cipherText)
|
|
|
|
|
|
|
|
|
|
|
|
// 发起 HTTP 请求
|
2025-08-25 15:44:06 +08:00
|
|
|
|
client := &http.Client{
|
|
|
|
|
|
Timeout: 20 * time.Second,
|
|
|
|
|
|
}
|
|
|
|
|
|
req, err := http.NewRequestWithContext(ctx, "POST", y.config.URL, strings.NewReader(content))
|
2025-07-28 01:46:39 +08:00
|
|
|
|
if err != nil {
|
2025-08-25 15:44:06 +08:00
|
|
|
|
err = fmt.Errorf("%w: %s", ErrSystem, err.Error())
|
|
|
|
|
|
if y.logger != nil {
|
|
|
|
|
|
y.logger.LogError(requestID, code, err, y.buildLogData(params, transactionID))
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil, err
|
2025-07-28 01:46:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
req.Header.Set("Content-Type", "application/json")
|
|
|
|
|
|
req.Header.Set("ACCT_ID", y.config.AcctID)
|
|
|
|
|
|
|
|
|
|
|
|
// 执行请求
|
|
|
|
|
|
resp, err := client.Do(req)
|
|
|
|
|
|
if err != nil {
|
2025-08-25 15:44:06 +08:00
|
|
|
|
err = fmt.Errorf("%w: %s", ErrSystem, err.Error())
|
|
|
|
|
|
if y.logger != nil {
|
|
|
|
|
|
y.logger.LogError(requestID, code, err, y.buildLogData(params, transactionID))
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil, err
|
2025-07-28 01:46:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
|
|
|
|
// 读取响应体
|
|
|
|
|
|
body, err := io.ReadAll(resp.Body)
|
|
|
|
|
|
if err != nil {
|
2025-08-25 15:44:06 +08:00
|
|
|
|
if y.logger != nil {
|
|
|
|
|
|
y.logger.LogError(requestID, code, err, y.buildLogData(params, transactionID))
|
|
|
|
|
|
}
|
2025-07-28 01:46:39 +08:00
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var respData []byte
|
|
|
|
|
|
|
2025-08-03 19:18:53 +08:00
|
|
|
|
if IsJSON(string(body)) {
|
|
|
|
|
|
respData = body
|
|
|
|
|
|
} else {
|
|
|
|
|
|
sDec, err := base64.StdEncoding.DecodeString(string(body))
|
|
|
|
|
|
if err != nil {
|
2025-08-25 15:44:06 +08:00
|
|
|
|
err = fmt.Errorf("%w: %s", ErrSystem, err.Error())
|
|
|
|
|
|
if y.logger != nil {
|
|
|
|
|
|
y.logger.LogError(requestID, code, err, y.buildLogData(params, transactionID))
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil, err
|
2025-08-03 19:18:53 +08:00
|
|
|
|
}
|
|
|
|
|
|
respData = y.AES_CBC_Decrypt(sDec, key)
|
2025-07-28 01:46:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
retCode := gjson.GetBytes(respData, "retcode").String()
|
|
|
|
|
|
|
2025-08-25 15:44:06 +08:00
|
|
|
|
// 记录响应日志
|
|
|
|
|
|
if y.logger != nil {
|
|
|
|
|
|
duration := time.Since(startTime)
|
|
|
|
|
|
y.logger.LogResponse(requestID, code, resp.StatusCode, respData, duration)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-28 01:46:39 +08:00
|
|
|
|
if retCode == "100000" {
|
|
|
|
|
|
// retcode 为 100000,表示查询为空
|
|
|
|
|
|
return nil, ErrNotFound
|
|
|
|
|
|
} else if retCode == "000000" {
|
|
|
|
|
|
// retcode 为 000000,表示有数据,返回 retdata
|
|
|
|
|
|
retData := gjson.GetBytes(respData, "retdata")
|
|
|
|
|
|
if !retData.Exists() {
|
2025-08-25 15:44:06 +08:00
|
|
|
|
err = fmt.Errorf("%w: %s", ErrDatasource, "羽山请求retdata为空")
|
|
|
|
|
|
if y.logger != nil {
|
|
|
|
|
|
y.logger.LogError(requestID, code, err, y.buildLogData(params, transactionID))
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil, err
|
2025-07-28 01:46:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
return []byte(retData.Raw), nil
|
|
|
|
|
|
} else {
|
2025-08-25 15:44:06 +08:00
|
|
|
|
err = fmt.Errorf("%w: %s", ErrDatasource, "羽山请求未知的状态码")
|
|
|
|
|
|
if y.logger != nil {
|
|
|
|
|
|
y.logger.LogError(requestID, code, err, y.buildLogData(params, transactionID))
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil, err
|
2025-07-28 01:46:39 +08:00
|
|
|
|
}
|
2025-08-25 15:44:06 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// generateRequestID 生成请求ID
|
|
|
|
|
|
func (y *YushanService) generateRequestID() string {
|
|
|
|
|
|
timestamp := time.Now().UnixNano()
|
|
|
|
|
|
hash := md5.Sum([]byte(fmt.Sprintf("%d_%s", timestamp, y.config.ApiKey)))
|
|
|
|
|
|
return fmt.Sprintf("yushan_%x", hash[:8])
|
|
|
|
|
|
}
|
2025-07-28 01:46:39 +08:00
|
|
|
|
|
2025-08-25 15:44:06 +08:00
|
|
|
|
// buildLogData 构建包含transactionId的日志数据
|
|
|
|
|
|
func (y *YushanService) buildLogData(data map[string]interface{}, transactionID string) map[string]interface{} {
|
|
|
|
|
|
if transactionID == "" {
|
|
|
|
|
|
return data
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
logData := data
|
|
|
|
|
|
if logData == nil {
|
|
|
|
|
|
logData = make(map[string]interface{})
|
|
|
|
|
|
}
|
|
|
|
|
|
logData["transaction_id"] = transactionID
|
|
|
|
|
|
return logData
|
2025-07-28 01:46:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// GenerateRandomString 生成一个32位的随机字符串订单号
|
|
|
|
|
|
func (y *YushanService) GenerateRandomString() (string, error) {
|
|
|
|
|
|
// 创建一个16字节的数组
|
|
|
|
|
|
bytes := make([]byte, 16)
|
|
|
|
|
|
// 读取随机字节到数组中
|
|
|
|
|
|
if _, err := rand.Read(bytes); err != nil {
|
|
|
|
|
|
return "", err
|
|
|
|
|
|
}
|
|
|
|
|
|
// 将字节数组编码为16进制字符串
|
|
|
|
|
|
return hex.EncodeToString(bytes), nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// AEC加密(CBC模式)
|
|
|
|
|
|
func (y *YushanService) AES_CBC_Encrypt(plainText []byte, key []byte) []byte {
|
|
|
|
|
|
//指定加密算法,返回一个AES算法的Block接口对象
|
|
|
|
|
|
block, err := aes.NewCipher(key)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
panic(err)
|
|
|
|
|
|
}
|
|
|
|
|
|
//进行填充
|
|
|
|
|
|
plainText = Padding(plainText, block.BlockSize())
|
|
|
|
|
|
//指定初始向量vi,长度和block的块尺寸一致
|
|
|
|
|
|
iv := []byte("0000000000000000")
|
|
|
|
|
|
//指定分组模式,返回一个BlockMode接口对象
|
|
|
|
|
|
blockMode := cipher.NewCBCEncrypter(block, iv)
|
|
|
|
|
|
//加密连续数据库
|
|
|
|
|
|
cipherText := make([]byte, len(plainText))
|
|
|
|
|
|
blockMode.CryptBlocks(cipherText, plainText)
|
|
|
|
|
|
//返回base64密文
|
|
|
|
|
|
return cipherText
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// AEC解密(CBC模式)
|
|
|
|
|
|
func (y *YushanService) AES_CBC_Decrypt(cipherText []byte, key []byte) []byte {
|
|
|
|
|
|
//指定解密算法,返回一个AES算法的Block接口对象
|
|
|
|
|
|
block, err := aes.NewCipher(key)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
panic(err)
|
|
|
|
|
|
}
|
|
|
|
|
|
//指定初始化向量IV,和加密的一致
|
|
|
|
|
|
iv := []byte("0000000000000000")
|
|
|
|
|
|
//指定分组模式,返回一个BlockMode接口对象
|
|
|
|
|
|
blockMode := cipher.NewCBCDecrypter(block, iv)
|
|
|
|
|
|
//解密
|
|
|
|
|
|
plainText := make([]byte, len(cipherText))
|
|
|
|
|
|
blockMode.CryptBlocks(plainText, cipherText)
|
|
|
|
|
|
//删除填充
|
|
|
|
|
|
plainText = UnPadding(plainText)
|
|
|
|
|
|
return plainText
|
|
|
|
|
|
} // 对明文进行填充
|
|
|
|
|
|
func Padding(plainText []byte, blockSize int) []byte {
|
|
|
|
|
|
//计算要填充的长度
|
|
|
|
|
|
n := blockSize - len(plainText)%blockSize
|
|
|
|
|
|
//对原来的明文填充n个n
|
|
|
|
|
|
temp := bytes.Repeat([]byte{byte(n)}, n)
|
|
|
|
|
|
plainText = append(plainText, temp...)
|
|
|
|
|
|
return plainText
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 对密文删除填充
|
|
|
|
|
|
func UnPadding(cipherText []byte) []byte {
|
|
|
|
|
|
//取出密文最后一个字节end
|
|
|
|
|
|
end := cipherText[len(cipherText)-1]
|
|
|
|
|
|
//删除填充
|
|
|
|
|
|
cipherText = cipherText[:len(cipherText)-int(end)]
|
|
|
|
|
|
return cipherText
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 判断字符串是否为 JSON 格式
|
|
|
|
|
|
func IsJSON(s string) bool {
|
|
|
|
|
|
var js interface{}
|
|
|
|
|
|
return json.Unmarshal([]byte(s), &js) == nil
|
|
|
|
|
|
}
|