package pay import ( "bytes" "encoding/xml" "errors" "sort" "strconv" "github.com/silenceper/wechat/context" "github.com/silenceper/wechat/util" ) var payGateway = "https://api.mch.weixin.qq.com/pay/unifiedorder" // Pay struct extends context type Pay struct { *context.Context } // Params was NEEDED when request unifiedorder // 传入的参数,用于生成 prepay_id 的必需参数 type Params struct { TotalFee string CreateIP string Body string OutTradeNo string OpenID string TradeType string } // Config 是传出用于 jsdk 用的参数 type Config struct { Timestamp int64 NonceStr string PrePayID string SignType string Sign string } // PreOrder 是 unifie order 接口的返回 type PreOrder struct { ReturnCode string `xml:"return_code"` ReturnMsg string `xml:"return_msg"` AppID string `xml:"appid,omitempty"` MchID string `xml:"mch_id,omitempty"` NonceStr string `xml:"nonce_str,omitempty"` Sign string `xml:"sign,omitempty"` ResultCode string `xml:"result_code,omitempty"` TradeType string `xml:"trade_type,omitempty"` PrePayID string `xml:"prepay_id,omitempty"` CodeURL string `xml:"code_url,omitempty"` ErrCode string `xml:"err_code,omitempty"` ErrCodeDes string `xml:"err_code_des,omitempty"` } //payRequest 接口请求参数 type payRequest struct { AppID string `xml:"appid"` MchID string `xml:"mch_id"` DeviceInfo string `xml:"device_info,omitempty"` NonceStr string `xml:"nonce_str"` Sign string `xml:"sign"` SignType string `xml:"sign_type,omitempty"` Body string `xml:"body"` Detail string `xml:"detail,omitempty"` Attach string `xml:"attach,omitempty"` //附加数据 OutTradeNo string `xml:"out_trade_no"` //商户订单号 FeeType string `xml:"fee_type,omitempty"` //标价币种 TotalFee string `xml:"total_fee"` //标价金额 SpbillCreateIP string `xml:"spbill_create_ip"` //终端IP TimeStart string `xml:"time_start,omitempty"` //交易起始时间 TimeExpire string `xml:"time_expire,omitempty"` //交易结束时间 GoodsTag string `xml:"goods_tag,omitempty"` //订单优惠标记 NotifyURL string `xml:"notify_url"` //通知地址 TradeType string `xml:"trade_type"` //交易类型 ProductID string `xml:"product_id,omitempty"` //商品ID LimitPay string `xml:"limit_pay,omitempty"` // OpenID string `xml:"openid,omitempty"` //用户标识 SceneInfo string `xml:"scene_info,omitempty"` //场景信息 } // NewPay return an instance of Pay package func NewPay(ctx *context.Context) *Pay { pay := Pay{Context: ctx} return &pay } // PrePayOrder return data for invoke wechat payment func (pcf *Pay) PrePayOrder(p *Params) (payOrder PreOrder, err error) { nonceStr := util.RandomStr(32) param := make(map[string]interface{}) param["appid"] = pcf.AppID param["body"] = p.Body param["mch_id"] = pcf.PayMchID param["nonce_str"] =nonceStr param["notify_url"] =pcf.PayNotifyURL param["out_trade_no"] =p.OutTradeNo param["spbill_create_ip"] =p.CreateIP param["total_fee"] =p.TotalFee param["trade_type"] =p.TradeType param["openid"] = p.OpenID bizKey := "&key="+pcf.PayKey str := orderParam(param,bizKey) sign := util.MD5Sum(str) request := payRequest{ AppID: pcf.AppID, MchID: pcf.PayMchID, NonceStr: nonceStr, Sign: sign, Body: p.Body, OutTradeNo: p.OutTradeNo, TotalFee: p.TotalFee, SpbillCreateIP: p.CreateIP, NotifyURL: pcf.PayNotifyURL, TradeType: p.TradeType, OpenID: p.OpenID, } rawRet, err := util.PostXML(payGateway, request) if err != nil { return } err = xml.Unmarshal(rawRet, &payOrder) if err != nil { return } if payOrder.ReturnCode == "SUCCESS" { //pay success if payOrder.ResultCode == "SUCCESS" { err = nil return } err = errors.New(payOrder.ErrCode + payOrder.ErrCodeDes) return } err = errors.New("[msg : xmlUnmarshalError] [rawReturn : " + string(rawRet) + "] [params : " + str + "] [sign : " + sign + "]") return } // PrePayID will request wechat merchant api and request for a pre payment order id func (pcf *Pay) PrePayID(p *Params) (prePayID string, err error) { order, err := pcf.PrePayOrder(p) if err != nil { return } if order.PrePayID == "" { err = errors.New("empty prepayid") } prePayID = order.PrePayID return } // order params func orderParam(source interface{}, bizKey string) (returnStr string) { switch v := source.(type) { case map[string]string: keys := make([]string, 0, len(v)) for k := range v { if k == "sign" { continue } keys = append(keys, k) } sort.Strings(keys) var buf bytes.Buffer for _, k := range keys { if v[k] == "" { continue } if buf.Len() > 0 { buf.WriteByte('&') } buf.WriteString(k) buf.WriteByte('=') buf.WriteString(v[k]) } buf.WriteString(bizKey) returnStr = buf.String() case map[string]interface{}: keys := make([]string, 0, len(v)) for k := range v { if k == "sign" { continue } keys = append(keys, k) } sort.Strings(keys) var buf bytes.Buffer for _, k := range keys { if v[k] == "" { continue } if buf.Len() > 0 { buf.WriteByte('&') } buf.WriteString(k) buf.WriteByte('=') switch vv := v[k].(type) { case string: buf.WriteString(vv) case int: buf.WriteString(strconv.FormatInt(int64(vv), 10)) default: panic("params type not supported") } } buf.WriteString(bizKey) returnStr = buf.String() } return }