| 
									
										
										
										
											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 { | 
					
						
							| 
									
										
										
										
											2025-08-26 16:03:46 +08:00
										 |  |  |  | 		y.logger.LogRequest(requestID, transactionID, code, y.config.URL, params) | 
					
						
							| 
									
										
										
										
											2025-08-25 15:44:06 +08:00
										 |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 { | 
					
						
							| 
									
										
										
										
											2025-08-26 16:03:46 +08:00
										 |  |  |  | 			y.logger.LogError(requestID, transactionID, code, err, params) | 
					
						
							| 
									
										
										
										
											2025-08-25 15:44:06 +08:00
										 |  |  |  | 		} | 
					
						
							|  |  |  |  | 		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 { | 
					
						
							| 
									
										
										
										
											2025-08-26 16:03:46 +08:00
										 |  |  |  | 			y.logger.LogError(requestID, transactionID, code, err, params) | 
					
						
							| 
									
										
										
										
											2025-08-25 15:44:06 +08:00
										 |  |  |  | 		} | 
					
						
							|  |  |  |  | 		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 { | 
					
						
							| 
									
										
										
										
											2025-08-26 16:03:46 +08:00
										 |  |  |  | 			y.logger.LogError(requestID, transactionID, code, err, params) | 
					
						
							| 
									
										
										
										
											2025-08-25 15:44:06 +08:00
										 |  |  |  | 		} | 
					
						
							|  |  |  |  | 		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 { | 
					
						
							| 
									
										
										
										
											2025-08-26 16:03:46 +08:00
										 |  |  |  | 			y.logger.LogError(requestID, transactionID, code, err, params) | 
					
						
							| 
									
										
										
										
											2025-08-25 15:44:06 +08:00
										 |  |  |  | 		} | 
					
						
							|  |  |  |  | 		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 { | 
					
						
							| 
									
										
										
										
											2025-08-26 16:03:46 +08:00
										 |  |  |  | 			y.logger.LogError(requestID, transactionID, code, err, params) | 
					
						
							| 
									
										
										
										
											2025-08-25 15:44:06 +08:00
										 |  |  |  | 		} | 
					
						
							| 
									
										
										
										
											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 { | 
					
						
							| 
									
										
										
										
											2025-08-26 16:03:46 +08:00
										 |  |  |  | 				y.logger.LogError(requestID, transactionID, code, err, params) | 
					
						
							| 
									
										
										
										
											2025-08-25 15:44:06 +08:00
										 |  |  |  | 			} | 
					
						
							|  |  |  |  | 			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) | 
					
						
							| 
									
										
										
										
											2025-08-26 16:03:46 +08:00
										 |  |  |  | 		y.logger.LogResponse(requestID, transactionID, code, resp.StatusCode, respData, duration) | 
					
						
							| 
									
										
										
										
											2025-08-25 15:44:06 +08:00
										 |  |  |  | 	} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 { | 
					
						
							| 
									
										
										
										
											2025-08-26 16:03:46 +08:00
										 |  |  |  | 				y.logger.LogError(requestID, transactionID, code, err, params) | 
					
						
							| 
									
										
										
										
											2025-08-25 15:44:06 +08:00
										 |  |  |  | 			} | 
					
						
							|  |  |  |  | 			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 { | 
					
						
							| 
									
										
										
										
											2025-08-26 16:03:46 +08:00
										 |  |  |  | 			y.logger.LogError(requestID, transactionID, code, err, params) | 
					
						
							| 
									
										
										
										
											2025-08-25 15:44:06 +08:00
										 |  |  |  | 		} | 
					
						
							|  |  |  |  | 		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
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | // 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 | 
					
						
							|  |  |  |  | } |