当前位置: 移动技术网 > IT编程>开发语言>PHP > 微信支付的开发流程详解

微信支付的开发流程详解

2017年12月12日  | 移动技术网IT编程  | 我要评论

火力少年王4,幕后玩家在线观看,丝袜高跟脚

最近在公司做了微信支付的接入,这里总结下开发的一些经验

注意,我使用的是微信开放平台的支付,与手机app相关,而与公众账号无关。

微信支付的主要操作流程

1.用户浏览app,选定商品然后下单。

2.服务器处理订单逻辑,开始正式发起支付流程

3.首先,后台服务器向weixin服务器发起请求,获取一个token。

4.后台服务器拿到token,使用和其他参数加密,再次向weixin服务器发起请求,获取一个预支付prepayid

5.后台服务器将该prepayid返回给app客户端

6.app调用手机上的微信控件,完成付款流程。

7.app向后台服务器发起一个回调请求,通知服务器交易完成。

8.weixin服务器处理完所有的流程后,向后台服务器发起一个post请求,正式通知后台服务器交易完毕

上面流程的一些注意点:

1.每次获取的token是有时效的,默认是7200s,而且每天最多获取200次,因此最好放到redis中缓存起来,等失效后再去重新获取

2.app发起的回调默认是不可靠的,后台应该尽可能(不是必须)向微信服务器发起订单查询,查询本次交易的结果。

3.weixin服务器向后台发起的notify,才是确保交易完成的最后屏障。后台服务器确认后必须返回“success”,否则weixin服务器会尝试重发请求。

获取token

这步很简单,发送一个get请求即可。只需配置正确参数。

‘‘‘从微信服务器获取token‘‘‘
  def _getaccesstokenfromweixin(self):
    response = requests.get(self.tokenurl % (self.appid, self.appsecret))
    if response.status_code == 200:
      text = response.text
      tokeninfo = json.loads(text)
      try:
        token = tokeninfo[‘access_token‘]
        expires_in = tokeninfo[‘expires_in‘]
        self._writeweixintokenlog(token, self.order_no)
        return token
      except keyerror:
        return none #token获取失败
    return none #http请求失败

获取prepayid

在微信支付的开发流程中,最繁琐的就是获取prepayid。

这一步我们需要组装这样一个参数:

{
"appid":"wxd930ea5d5a258f4f",
"traceid":"test_1399514976",
"noncestr":"e7d161ac8d8a76529d39d9f5b4249ccb ",
"timestamp":1399514976, "package":"bank_type=wx&body=%e6%94%af%e4%bb%98%e6%b5%8b%e8%af%
95&fee_type=1&input_charset=utf-8¬ify_url=http%3a%2f%2fweixin.qq.com&out_trade_ no=7240b65810859cbf2a8d9f76a638c0a3&partner=1900000109&spbill_create_ip=196.168.1.1& total_fee=1&sign=7f77b507b755b3262884291517e380f8",
"sign_method":"sha1", "app_signature":"7f77b507b755b3262884291517e380f8"
}
 

组装package

这里的第一步就是组装package:

"package":"bank_type=wx&body=%e6%94%af%e4%bb%98%e6%b5%8b%e8%af%
95&fee_type=1&input_charset=utf-8¬ify_url=http%3a%2f%2fweixin.qq.com&out_trade_ no=7240b65810859cbf2a8d9f76a638c0a3&partner=1900000109&spbill_create_ip=196.168.1.1& total_fee=1&sign=7f77b507b755b3262884291517e380f8",

组装package需要的参数如上面代码所示,所以我们需要准备一个params,然后准备签名,签名流程如下:

1.按照key的字典序,对params进行排序,然后拼接成字符串,注意这些key不包括sign

2.在上面的字符串后面拼接key=paternerkey,然后对整个字符串进行md5签名,然后转换成大写,此时我们就得到了签名

然后我们将所有params的value进行urlencode转码,然后后面拼接上sign=signvalue,就得到了package字符串。

这里创建md5的过如下:

def createmd5signature(self, signparams):
    ‘‘‘先排序‘‘‘
    sortedparams = sorted(signparams.iteritems(), key=lambda d:d[0])
    ‘‘‘拼接‘‘‘  
    stringsigntemp = "&".join(["%s=%s" % (item[0], item[1]) for item in sortedparams if item[0] != ‘sign‘ and ‘‘ != item[1]])
    #加上财付通商户权限密钥
    stringsigntemp += ‘&key=%s‘ % (self.partnerkey)
    #使用md5进行签名,然后转化为大写
    stringsign = hashlib.md5(stringsigntemp).hexdigest().upper()  #upper
    return stringsign

组装package的代码:

def getpackage(self, packageparams):
    ‘‘‘先获取params的sign,然后将params进行urlencode,最后拼接,加上sign‘‘‘
    sign = self.createmd5signature(packageparams)
    packageparams = sorted(packageparams.iteritems(), key=lambda d:d[0])
    stringparams = "&".join(["%s=%s" % (item[0], urllib.quote(str(item[1]))) for item in packageparams])
    stringparams += ‘&sign=%s‘ % (sign)
    return stringparams

继续组装参数

得到package后,我们继续组装参数:

这里需要的参数为:

appid=wxd930ea5d5a258f4f
appkey=l8lrmqqegrxst5reoub0k66cay a wpqha vsq7ggkkxhcoastwksvux1uvmvqcl xahoyd3elnbrno2dhnnzgfvg9qs473m3dtozug5er46fhugofumv8h2fvr9qkjslc5k
noncestr=e7d161ac8d8a76529d39d9f5b4249ccb
package=bank_type=wx&body=%e6%94%af%e4%bb%98%e6%b5%8b%e8%af%95 &fee_type=1&input_charset=utf-8¬ify_url=http%3a%2f%2fweixin.qq.com&out_trade_no =7240b65810859cbf2a8d9f76a638c0a3&partner=1900000109&spbill_create_ip=196.168.1.1&tot al_fee=1&sign=7f77b507b755b3262884291517e380f8
timestamp=1399514976 

traceid=test_1399514976

注意这里有个坑:

参与签名的是上面的参数,但是最后的参数中不包括appkey,签名后要记得删除。

1.所有参数按照字典序排序,然后拼接

2.进行sha1签名,拼接到上面字符串的后面

3.注意这里要删除appkey,然后加上sign

获取sha1签名的代码如下:

def createsha1signature(self, params):
    ‘‘‘先排序,然后拼接‘‘‘
    sortedparams = sorted(params.iteritems(), key=lambda d:d[0]) 
    stringsigntemp = "&".join(["%s=%s" % (item[0], item[1]) for item in sortedparams])
    stringsign = hashlib.sha1(stringsigntemp).hexdigest()
    return stringsign

随后我们获取到这样的参数:

{
"appid":"wxd930ea5d5a258f4f", 
"noncestr":"e7d161ac8d8a76529d39d9f5b4249ccb", 
"package":"sign=wxpay";
"partnerid":"1900000109" 
"prepayid":"1101000000140429eb40476f8896f4c9", 
"sign":"7ffecb600d7157c5aa49810d2d8f28bc2811827b", 
"timestamp":"1399514976"
}

获取prepayid

代码如下:

‘‘‘获取预支付prepayid‘‘‘
  def gerprepayid(self, token, requestparams):
    ‘‘‘将参数,包括package,进行json化,然后发起post请求‘‘‘
    data = json.dumps(requestparams)
    response = requests.post(self.gateurl % (token), data=data)
    if response.status_code == 200:
      text = response.text
      text = json.loads(text)
      errcode = text[‘errcode‘]
      if errcode == 0:
        return text[‘prepayid‘]
    return none

我们获取的prepayid格式应该是这样:

{"prepayid":"1101000000140429eb40476f8896f4c9","errcode":0,"errmsg":"success"} 

再次签名

这里采用上面sha1的签名方式再次签名,获取到下面的参数:

{
"appid":"wxd930ea5d5a258f4f", 
"noncestr":"e7d161ac8d8a76529d39d9f5b4249ccb", 
"package":"sign=wxpay";
"partnerid":"1900000109" 
"prepayid":"1101000000140429eb40476f8896f4c9", 
"sign":"7ffecb600d7157c5aa49810d2d8f28bc2811827b", 
"timestamp":"1399514976"
}

后台服务器将该结果返回给app,此时app即可发起支付。

上面的流程代码为:

‘‘‘接收app的请求,返回prepayid‘‘‘
class weixinrequireprepaidhandler(basictemplatehandler):

  ‘‘‘这个方法在ordersaddhandler中被调用‘‘‘
  @staticmethod
  def getprepaidresult(order_no, total_pay, product_name, client_ip):
    ‘‘‘封装了常用的签名算法‘‘‘
    weixinrequesthandler = weixinrequesthandler(order_no)
    ‘‘‘收集订单相关信息‘‘‘
    addtion = str(random.randint(10, 100)) #产生一个两位的数字,拼接在订单号的后面
    out_trade_no = str(order_no) + addtion
    order_price = float(total_pay) #这里必须允许浮点数,后面转化成分之后转化为int
    #order_price = 0.01 #测试
    remote_addr = client_ip #客户端的ip地址
    print remote_addr
    current_time = int(time.time())
    order_create_time = str(current_time)
    order_deadline = str(current_time + 20*60)

    ‘‘‘这里的一些参数供下面使用‘‘‘
    noncestr = hashlib.md5(str(random.random())).hexdigest()
    timestamp = str(int(time.time()))
    pack = ‘sign=wxpay‘

    ‘‘‘获取token‘‘‘
    access_token = weixinrequesthandler.getaccesstoken()
    logging.info("get token: %s" % access_token)
    if access_token:
      ‘‘‘设置package参数‘‘‘
      packageparams = {}
      packageparams[‘bank_type‘] = ‘wx‘  #支付类型
      packageparams[‘body‘] = product_name #商品名称
      packageparams[‘fee_type‘] = ‘1‘   #人民币 fen
      packageparams[‘input_charset‘] = ‘gbk‘ #gbk
      packageparams[‘notify_url‘] = config[‘notify_url‘] #post异步消息通知
      packageparams[‘out_trade_no‘] = str(out_trade_no) #订单号
      packageparams[‘partner‘] = config[‘partnerid‘] #商户号
      packageparams[‘total_fee‘] = str(int(order_price*100))  #订单金额,单位是分
      packageparams[‘spbill_create_ip‘] = remote_addr #ip
      packageparams[‘time_start‘] = order_create_time #订单生成时间
      packageparams[‘time_expire‘] = order_deadline #订单失效时间

      ‘‘‘获取package‘‘‘
      package = weixinrequesthandler.getpackage(packageparams)

      ‘‘‘设置支付参数‘‘‘
      signparams = {}
      signparams[‘appid‘] = config[‘appid‘]
      signparams[‘appkey‘] = config[‘paysignkey‘] #delete
      signparams[‘noncestr‘] = noncestr
      signparams[‘package‘] = package
      signparams[‘timestamp‘] = timestamp
      signparams[‘traceid‘] = ‘mytraceid_001‘

      ‘‘‘生成支付签名‘‘‘
      app_signature = weixinrequesthandler.createsha1signature(signparams)
      ‘‘‘增加不参与签名的额外参数‘‘‘
      signparams[‘sign_method‘] = ‘sha1‘
      signparams[‘app_signature‘] = app_signature

      ‘‘‘剔除appkey‘‘‘
      del signparams[‘appkey‘]

      ‘‘‘获取prepayid‘‘‘
      prepayid = weixinrequesthandler.gerprepayid(access_token, signparams)
      if prepayid:

        ‘‘‘使用拿到的prepayid再次准备签名‘‘‘
        pack = ‘sign=wxpay‘
        prepayparams = {}
        prepayparams[‘appid‘] = config[‘appid‘]
        prepayparams[‘appkey‘] = config[‘paysignkey‘]
        prepayparams[‘noncestr‘] = noncestr
        prepayparams[‘package‘] = pack
        prepayparams[‘partnerid‘] = config[‘partnerid‘]
        prepayparams[‘prepayid‘] = prepayid
        prepayparams[‘timestamp‘] = timestamp

        ‘‘‘生成签名‘‘‘
        sign = weixinrequesthandler.createsha1signature(prepayparams)

        ‘‘‘准备输出参数‘‘‘
        returnparams = {}
        returnparams[‘status‘] = 0
        returnparams[‘retmsg‘] = ‘success‘
        returnparams[‘appid‘] = config[‘appid‘]
        returnparams[‘noncestr‘] = noncestr
        returnparams[‘package‘] = pack
        returnparams[‘prepayid‘] = prepayid
        returnparams[‘timestamp‘] = timestamp
        returnparams[‘sign‘] = sign
        returnparams[‘partnerid‘] = config[‘partnerid‘]
        returnparams[‘addtion‘] = addtion
        
      else:
        ‘‘‘prepayid获取失败‘‘‘
        returnparams = {}
        returnparams[‘status‘] = -1
        returnparams[‘retmsg‘] = ‘prepayid获取失败‘
    else: 
      ‘‘‘token获取失败‘‘‘
      returnparams = {}
      returnparams[‘status‘] = -1
      returnparams[‘retmsg‘] = ‘token获取失败‘

    ‘‘‘生成json格式文本,然后返回给app‘‘‘
    return returnparams

后台异步通知

微信服务器发来的notify异步通知,才是支付成功的最终标志,这一步处于安全起见,我们必须进行延签:

延签代码如下:

def istenpaysign(self, params):
    helper = weixinrequesthandler()
    sign = helper.createmd5signature(params)
    return params[‘sign‘] == sign

整体流程如下:

‘‘‘微信服务器向后台发送的异步通知‘‘‘
class weixinappnotifyhandler(basictemplatehandler):
  def initialize(self):
    self.weixinresponsehandler = weixinresponsehandler()

  def post(self):
    ‘‘‘解析参数‘‘‘
    params = self.parsequerystring()

    ‘‘‘验证是否是weixin服务器发回的消息‘‘‘
    verifyweixinsign = self.weixinresponsehandler.istenpaysign(params)
    ‘‘‘处理订单‘‘‘
    if verifyweixinsign:
      ‘‘‘订单逻辑‘‘‘
      order_no = str(params[‘out_trade_no‘])
      order_no = order_no[0:-2]
      print ‘%s paied successfully‘ % order_no
      self.saveweixinreceipt(params)
      updateorderspaidbyweixin(order_no) #更新订单使用状态
      consumecouponbyorderno(order_no) #优惠券已经使用
      self.write("success")
    else:
      self.write("fail")

  def parsequerystring(self):
    ‘‘‘获取url中所有的参数‘‘‘
    uri = self.request.uri
    ‘‘‘解析出uri中的query字符串‘‘‘
    parseresult = urlparse.urlparse(uri) 
    query = parseresult.query
    ‘‘‘解析query字符串‘‘‘
    params = urlparse.parse_qs(query)
    for item in params:
      params[item] = params[item][0].strip()
    return params

最后说明一点,用户在手机上付完款,并不算支付成功,只有weixin服务器收到notify通知返回的success时,才算交易最终成功,此时我们的手机可以收到微信官方发来的一条消息。

以上就是对微信支付开发流程的资料整理,后续继续补充相关资料,谢谢大家对本站的支持!

如对本文有疑问,请在下面进行留言讨论,广大热心网友会与你互动!! 点击进行留言回复

相关文章:

验证码:
移动技术网