123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242 |
- package alipay
- import (
- "crypto"
- "crypto/rand"
- "crypto/rsa"
- "crypto/sha1"
- "crypto/sha256"
- "crypto/x509"
- "encoding/base64"
- "encoding/pem"
- "errors"
- "fmt"
- "hash"
- "log"
- "net/url"
- "time"
- "github.com/iGoogle-ink/gopay"
- )
- // AppId string `json:"app_id"` //支付宝分配给开发者的应用ID
- // Method string `json:"method"` //接口名称
- // Format string `json:"format"` //仅支持 JSON
- // ReturnUrl string `json:"return_url"` //HTTP/HTTPS开头字符串
- // Charset string `json:"charset"` //请求使用的编码格式,如utf-8,gbk,gb2312等,推荐使用 utf-8
- // SignType string `json:"sign_type"` //商户生成签名字符串所使用的签名算法类型,目前支持RSA2和RSA,推荐使用 RSA2
- // Sign string `json:"sign"` //商户请求参数的签名串
- // Timestamp string `json:"timestamp"` //发送请求的时间,格式"yyyy-MM-dd HH:mm:ss"
- // Version string `json:"version"` //调用的接口版本,固定为:1.0
- // NotifyUrl string `json:"notify_url"` //支付宝服务器主动通知商户服务器里指定的页面http/https路径。
- // BizContent string `json:"biz_content"` //业务请求参数的集合,最大长度不限,除公共参数外所有请求参数都必须放在这个参数中传递,具体参照各产品快速接入文档
- type OpenApiRoyaltyDetailInfoPojo struct {
- RoyaltyType string `json:"royalty_type,omitempty"`
- TransOut string `json:"trans_out,omitempty"`
- TransOutType string `json:"trans_out_type,omitempty"`
- TransInType string `json:"trans_in_type,omitempty"`
- TransIn string `json:"trans_in"`
- Amount string `json:"amount,omitempty"`
- Desc string `json:"desc,omitempty"`
- }
- // 设置 支付宝 私钥类型,alipay.PKCS1 或 alipay.PKCS8,默认 PKCS1
- func (a *Client) SetPrivateKeyType(t PKCSType) (client *Client) {
- a.mu.Lock()
- a.PrivateKeyType = t
- a.mu.Unlock()
- return a
- }
- // 设置 时区,不设置或出错均为默认服务器时间
- func (a *Client) SetLocation(name string) (client *Client) {
- location, err := time.LoadLocation(name)
- if err != nil {
- log.Println("set Location err, default UTC")
- return a
- }
- a.mu.Lock()
- a.LocationName = name
- a.location = location
- a.mu.Unlock()
- return a
- }
- // 设置 应用公钥证书SN
- // appCertSN:应用公钥证书SN,通过 alipay.GetCertSN() 获取
- func (a *Client) SetAppCertSN(appCertSN string) (client *Client) {
- a.mu.Lock()
- a.AppCertSN = appCertSN
- a.mu.Unlock()
- return a
- }
- // 设置 支付宝公钥证书SN
- // aliPayPublicCertSN:支付宝公钥证书SN,通过 alipay.GetCertSN() 获取
- func (a *Client) SetAliPayPublicCertSN(aliPayPublicCertSN string) (client *Client) {
- a.mu.Lock()
- a.AliPayPublicCertSN = aliPayPublicCertSN
- a.mu.Unlock()
- return a
- }
- // 设置 支付宝CA根证书SN
- // aliPayRootCertSN:支付宝CA根证书SN,通过 alipay.GetRootCertSN() 获取
- func (a *Client) SetAliPayRootCertSN(aliPayRootCertSN string) (client *Client) {
- a.mu.Lock()
- a.AliPayRootCertSN = aliPayRootCertSN
- a.mu.Unlock()
- return a
- }
- // 设置 app_cert_sn、alipay_root_cert_sn、alipay_cert_sn 通过应用公钥证书路径
- // appCertPath:应用公钥证书路径
- // aliPayRootCertPath:支付宝根证书文件路径
- // aliPayPublicCertPath:支付宝公钥证书文件路径
- func (a *Client) SetCertSnByPath(appCertPath, aliPayRootCertPath, aliPayPublicCertPath string) (err error) {
- appCertSn, err := GetCertSN(appCertPath)
- if err != nil {
- return fmt.Errorf("get app_cert_sn return err, but alse return alipay client. err: %w", err)
- }
- rootCertSn, err := GetRootCertSN(aliPayRootCertPath)
- if err != nil {
- return fmt.Errorf("get alipay_root_cert_sn return err, but alse return alipay client. err: %w", err)
- }
- publicCertSn, err := GetCertSN(aliPayPublicCertPath)
- if err != nil {
- return fmt.Errorf("get alipay_cert_sn return err, but alse return alipay client. err: %w", err)
- }
- a.mu.Lock()
- a.AppCertSN = appCertSn
- a.AliPayRootCertSN = rootCertSn
- a.AliPayPublicCertSN = publicCertSn
- a.mu.Unlock()
- return nil
- }
- // 设置支付后的ReturnUrl
- func (a *Client) SetReturnUrl(url string) (client *Client) {
- a.mu.Lock()
- a.ReturnUrl = url
- a.mu.Unlock()
- return a
- }
- // 设置支付宝服务器主动通知商户服务器里指定的页面http/https路径。
- func (a *Client) SetNotifyUrl(url string) (client *Client) {
- a.mu.Lock()
- a.NotifyUrl = url
- a.mu.Unlock()
- return a
- }
- // 设置编码格式,如utf-8,gbk,gb2312等,默认推荐使用 utf-8
- func (a *Client) SetCharset(charset string) (client *Client) {
- a.mu.Lock()
- if charset == gopay.NULL {
- a.Charset = "utf-8"
- } else {
- a.Charset = charset
- }
- a.mu.Unlock()
- return a
- }
- // 设置签名算法类型,目前支持RSA2和RSA,默认推荐使用 RSA2
- func (a *Client) SetSignType(signType string) (client *Client) {
- a.mu.Lock()
- if signType == gopay.NULL {
- a.SignType = RSA2
- } else {
- a.SignType = signType
- }
- a.mu.Unlock()
- return a
- }
- // 设置应用授权
- func (a *Client) SetAppAuthToken(appAuthToken string) (client *Client) {
- a.mu.Lock()
- a.AppAuthToken = appAuthToken
- a.mu.Unlock()
- return a
- }
- // 设置用户信息授权
- func (a *Client) SetAuthToken(authToken string) (client *Client) {
- a.mu.Lock()
- a.AuthToken = authToken
- a.mu.Unlock()
- return a
- }
- // 获取支付宝参数签名
- // bm:签名参数
- // signType:签名类型,alipay.RSA 或 alipay.RSA2
- // t:私钥类型,alipay.PKCS1 或 alipay.PKCS1,默认 PKCS1
- // privateKey:应用私钥,支持PKCS1和PKCS8
- func GetRsaSign(bm gopay.BodyMap, signType string, t PKCSType, privateKey string) (sign string, err error) {
- var (
- block *pem.Block
- h hash.Hash
- key *rsa.PrivateKey
- hashs crypto.Hash
- encryptedBytes []byte
- )
- pk := FormatPrivateKey(privateKey)
- if block, _ = pem.Decode([]byte(pk)); block == nil {
- return gopay.NULL, errors.New("pem.Decode:privateKey decode error")
- }
- switch t {
- case PKCS1:
- if key, err = x509.ParsePKCS1PrivateKey(block.Bytes); err != nil {
- return gopay.NULL, err
- }
- case PKCS8:
- pkcs8Key, err := x509.ParsePKCS8PrivateKey(block.Bytes)
- if err != nil {
- return gopay.NULL, err
- }
- pk8, ok := pkcs8Key.(*rsa.PrivateKey)
- if !ok {
- return gopay.NULL, errors.New("parse PKCS8 key error")
- }
- key = pk8
- default:
- if key, err = x509.ParsePKCS1PrivateKey(block.Bytes); err != nil {
- return gopay.NULL, err
- }
- }
- switch signType {
- case RSA:
- h = sha1.New()
- hashs = crypto.SHA1
- case RSA2:
- h = sha256.New()
- hashs = crypto.SHA256
- default:
- h = sha256.New()
- hashs = crypto.SHA256
- }
- if _, err = h.Write([]byte(bm.EncodeAliPaySignParams())); err != nil {
- return
- }
- if encryptedBytes, err = rsa.SignPKCS1v15(rand.Reader, key, hashs, h.Sum(nil)); err != nil {
- return
- }
- sign = base64.StdEncoding.EncodeToString(encryptedBytes)
- return
- }
- // 格式化请求URL参数
- func FormatURLParam(body gopay.BodyMap) (urlParam string) {
- v := url.Values{}
- for key, value := range body {
- v.Add(key, value.(string))
- }
- return v.Encode()
- }
|