代码拉取完成,页面将自动刷新
同步操作将从 springrain/gowe 强制同步,此操作会覆盖自 Fork 仓库以来所做的任何修改,且无法恢复!!!
确定后同步将在后台操作,完成时将刷新页面,请耐心等待。
package gowe
import (
"bytes"
"crypto/tls"
"encoding/json"
"encoding/pem"
"fmt"
"io/ioutil"
"net"
"net/http"
"net/url"
"regexp"
"strings"
"time"
"gitee.com/chunanyong/gouuid"
"golang.org/x/crypto/pkcs12"
)
//http请求的client
var client *http.Client
//初始化 http连接信息
func init() {
client = &http.Client{
Timeout: 30 * time.Second,
Transport: &http.Transport{
IdleConnTimeout: 3 * time.Minute,
TLSHandshakeTimeout: 10 * time.Second,
DialContext: (&net.Dialer{
Timeout: 30 * time.Second,
KeepAlive: 10 * time.Minute,
DualStack: true,
}).DialContext,
},
}
}
//httpGet 发起get请求
func httpGet(apiurl string) ([]byte, error) {
resp, errGet := client.Get(apiurl)
if errGet != nil {
return nil, errGet
}
defer resp.Body.Close()
body, errRead := ioutil.ReadAll(resp.Body)
return body, errRead
}
//httpPost post请求,返回原始的字节数组
func httpPost(apiurl string, params map[string]interface{}) ([]byte, error) {
//data := make(url.Values)
//for k, v := range params {
// data.Add(k, v)
//}
byteparams, errparams := json.Marshal(params)
if errparams != nil {
return nil, errparams
}
resp, errPost := client.Post(apiurl, "application/json", bytes.NewReader(byteparams))
if errPost != nil {
return nil, errPost
}
defer resp.Body.Close()
body, errRead := ioutil.ReadAll(resp.Body)
return body, errRead
}
//httpPostXml 发送Post请求,参数是XML格式的字符串
func httpPostXml(url string, xmlBody string) (body []byte, err error) {
resp, err := client.Post(url, "application/xml", strings.NewReader(xmlBody))
if err != nil {
return
}
defer resp.Body.Close()
body, err = ioutil.ReadAll(resp.Body)
return
}
//httpPostXmlWithCert 发送带证书的Post请求,参数是XML格式的字符串
func httpPostXmlWithCert(url string, xmlBody string, client *http.Client) (body []byte, err error) {
resp, err := client.Post(url, "application/xml", strings.NewReader(xmlBody))
if err != nil {
return
}
defer resp.Body.Close()
body, err = ioutil.ReadAll(resp.Body)
return
}
// mchType 0:普通商户接口, 1:特殊的商户接口(企业付款,微信找零),2:红包
func wxPayBuildBody(wxPayConfig IWxPayConfig, bodyObj interface{}, mchType int) (body map[string]interface{}, err error) {
// 将bodyObj转换为map[string]interface{}类型
bodyJson, _ := json.Marshal(bodyObj)
body = make(map[string]interface{})
_ = json.Unmarshal(bodyJson, &body)
// 添加固定参数
if mchType == 1 { //特殊的商户接口(企业付款,微信找零)
body["mch_appid"] = wxPayConfig.GetAppId()
body["mchid"] = wxPayConfig.GetMchId()
} else if mchType == 2 { //红包
body["wxappid"] = wxPayConfig.GetAppId() //微信分配的公众账号ID(企业号corpid即为此appId).接口传入的所有appid应该为公众号的appid(在mp.weixin.qq.com申请的),不能为APP的appid(在open.weixin.qq.com申请的)
body["mch_id"] = wxPayConfig.GetMchId()
} else { //普通微信支付
body["appid"] = wxPayConfig.GetAppId()
body["mch_id"] = wxPayConfig.GetMchId()
}
//如果是服务商模式
if isWxPayFacilitator(wxPayConfig.GetServiceType()) {
body["sub_appid"] = wxPayConfig.GetSubAppId()
body["sub_mch_id"] = wxPayConfig.GetSubMchId()
}
//nonceStr := getRandomString(32)
nonceStr := FuncGenerateRandomString()
body["nonce_str"] = nonceStr
// 生成签名
signType, _ := body["sign_type"].(string)
var sign string
if wxPayConfig.IsProd() {
sign = wxPayLocalSign(body, signType, wxPayConfig.GetAPIKey())
} else {
body["sign_type"] = SignTypeMD5
key, iErr := wxPaySandboxSign(wxPayConfig, nonceStr, SignTypeMD5)
if err = iErr; iErr != nil {
return
}
sign = wxPayLocalSign(body, SignTypeMD5, key)
}
body["sign"] = sign
return
}
// 是否是服务商模式
func isWxPayFacilitator(serviceType int) bool {
switch serviceType {
case ServiceTypeFacilitatorDomestic, ServiceTypeFacilitatorAbroad, ServiceTypeBankServiceProvidor:
return true
default:
return false
}
}
//generateXml 生成请求XML的Body体
func generateXml(data map[string]interface{}) string {
buffer := new(bytes.Buffer)
buffer.WriteString("<xml>")
for k, v := range data {
buffer.WriteString(fmt.Sprintf("<%s><![CDATA[%v]]></%s>", k, v, k))
}
buffer.WriteString("</xml>")
return buffer.String()
}
//jsonString 生成模型字符串
func jsonString(m interface{}) string {
bs, _ := json.Marshal(m)
return string(bs)
}
//formatDateTime 格式化时间,按照yyyyMMddHHmmss格式
func formatDateTime(t time.Time) string {
return t.Format("20060102150405")
}
//encodePath 对URL进行Encode编码
func encodePath(u string) (path string, err error) {
uriObj, err := url.Parse(u)
if err != nil {
return
}
path = uriObj.EscapedPath()
return
}
//pkcs7UnPadding 解密填充模式(去除补全码) pkcs7UnPadding 解密时,需要在最后面去掉加密时添加的填充byte
func pkcs7UnPadding(plainText []byte) []byte {
length := len(plainText)
unpadding := int(plainText[length-1]) // 找到Byte数组最后的填充byte
return plainText[:(length - unpadding)] // 只截取返回有效数字内的byte数组
}
//isValidAuthCode 18位纯数字,以10、11、12、13、14、15开头
func isValidAuthCode(authcode string) (ok bool) {
pattern := "^1[0-5][0-9]{16}$"
ok, _ = regexp.MatchString(pattern, authcode)
return
}
/*
//getRandomString 获取随机字符串
func getRandomString(length int) string {
str := "0123456789AaBbCcDdEeFfGgHhIiJjKkLlMmNnOoPpQqRrSsTtUuVvWwXxYyZz"
b := []byte(str)
var result []byte
r := rand.New(rand.NewSource(time.Now().UnixNano()))
for i := 0; i < length; i++ {
result = append(result, b[r.Intn(len(b))])
}
return string(result)
}
*/
//FuncGenerateRandomString 生成32位的随机字符串
var FuncGenerateRandomString func() string = generateRandomString
//generateRandomString 生成32位随机字符串
func generateRandomString() string {
pk, errUUID := gouuid.NewV4()
if errUUID != nil {
return ""
}
return strings.Replace(pk.String(), "-", "", -1)
}
//wxPayDoWeChat 向微信发送请求
func wxPayDoWeChat(wxPayConfig IWxPayConfig, apiuri string, bodyObj interface{}, mchType int) (bytes []byte, err error) {
apiurl := WxPayMchAPIURL + apiuri
if !wxPayConfig.IsProd() {
apiurl = WxPaySanBoxAPIURL + apiuri
}
// 转换参数
body, err := wxPayBuildBody(wxPayConfig, bodyObj, mchType)
if err != nil {
return
}
// 发起请求
bytes, err = httpPostXml(apiurl, generateXml(body))
return
}
//wxPayDoWeChatWithCert 向微信发送带证书请求
// mchType 0:普通商户接口, 1:特殊的商户接口(企业付款,微信找零),2:红包
func wxPayDoWeChatWithCert(wxPayConfig IWxPayConfig, apiuri string, bodyObj interface{}, mchType int) ([]byte, error) {
// 转换参数
body, err := wxPayBuildBody(wxPayConfig, bodyObj, mchType)
if err != nil {
return nil, err
}
// 设置证书和连接池
client, err := wxPayGetCertHttpClient(wxPayConfig)
if err != nil {
return nil, err
}
apiurl := WxPayMchAPIURL + apiuri
if !wxPayConfig.IsProd() {
apiurl = WxPaySanBoxAPIURL + apiuri
}
// 发起请求
bytes, err := httpPostXmlWithCert(apiurl, generateXml(body), client)
return bytes, err
}
//wxPayGetCertHttpClient 获取带证数的httpClient
func wxPayGetCertHttpClient(wxPayConfig IWxPayConfig) (*http.Client, error) {
certPath := wxPayConfig.GetCertificateFile()
certData, err := ioutil.ReadFile(certPath)
if err != nil {
return nil, err
}
client, err := wxPayBuildClient(certData, wxPayConfig.GetMchId())
return client, err
}
//wxPayBuildClient 构建带证数的httpClient
func wxPayBuildClient(data []byte, mchId string) (client *http.Client, err error) {
// 将pkcs12证书转成pem
cert, err := wxPayPkc12ToPerm(data, mchId)
if err != nil {
return
}
// tls配置
config := &tls.Config{Certificates: []tls.Certificate{cert}}
// 带证书的客户端
client = &http.Client{
Timeout: 30 * time.Second,
Transport: &http.Transport{
IdleConnTimeout: 3 * time.Minute,
TLSHandshakeTimeout: 10 * time.Second,
DialContext: (&net.Dialer{
Timeout: 30 * time.Second,
KeepAlive: 10 * time.Minute,
DualStack: true,
}).DialContext,
TLSClientConfig: config,
DisableCompression: true,
},
}
return
}
//wxPayPkc12ToPerm 证数格式转化
func wxPayPkc12ToPerm(data []byte, mchId string) (cert tls.Certificate, err error) {
blocks, err := pkcs12.ToPEM(data, mchId)
if err != nil {
return
}
var pemData []byte
for _, b := range blocks {
pemData = append(pemData, pem.EncodeToMemory(b)...)
}
cert, err = tls.X509KeyPair(pemData, pemData)
return
}
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。