成人午夜视频全免费观看高清-秋霞福利视频一区二区三区-国产精品久久久久电影小说-亚洲不卡区三一区三区一区

微信支付開發(fā)流程的示例分析-創(chuàng)新互聯(lián)

這篇文章給大家分享的是有關(guān)微信支付開發(fā)流程的示例分析的內(nèi)容。小編覺得挺實用的,因此分享給大家做個參考,一起跟隨小編過來看看吧。

創(chuàng)新互聯(lián)是一家專業(yè)提供鶴慶企業(yè)網(wǎng)站建設(shè),專注與成都做網(wǎng)站、成都網(wǎng)站建設(shè)、成都h5網(wǎng)站建設(shè)、小程序制作等業(yè)務(wù)。10年已為鶴慶眾多企業(yè)、政府機(jī)構(gòu)等服務(wù)。創(chuàng)新互聯(lián)專業(yè)的建站公司優(yōu)惠進(jìn)行中。

注意,我使用的是微信開放平臺的支付,與手機(jī)app相關(guān),而與公眾賬號無關(guān)。

微信支付的主要操作流程

1.用戶瀏覽app,選定商品然后下單。

2.服務(wù)器處理訂單邏輯,開始正式發(fā)起支付流程

3.首先,后臺服務(wù)器向weixin服務(wù)器發(fā)起請求,獲取一個token。

4.后臺服務(wù)器拿到token,使用和其他參數(shù)加密,再次向weixin服務(wù)器發(fā)起請求,獲取一個預(yù)支付prepayid

5.后臺服務(wù)器將該prepayid返回給app客戶端

6.app調(diào)用手機(jī)上的微信控件,完成付款流程。

7.app向后臺服務(wù)器發(fā)起一個回調(diào)請求,通知服務(wù)器交易完成。

8.weixin服務(wù)器處理完所有的流程后,向后臺服務(wù)器發(fā)起一個post請求,正式通知后臺服務(wù)器交易完畢

上面流程的一些注意點:

1.每次獲取的token是有時效的,默認(rèn)是7200s,而且每天最多獲取200次,因此好放到redis中緩存起來,等失效后再去重新獲取

2.app發(fā)起的回調(diào)默認(rèn)是不可靠的,后臺應(yīng)該盡可能(不是必須)向微信服務(wù)器發(fā)起訂單查詢,查詢本次交易的結(jié)果。

3.weixin服務(wù)器向后臺發(fā)起的notify,才是確保交易完成的最后屏障。后臺服務(wù)器確認(rèn)后必須返回“success”,否則weixin服務(wù)器會嘗試重發(fā)請求。

獲取token

這步很簡單,發(fā)送一個get請求即可。只需配置正確參數(shù)。

‘‘‘從微信服務(wù)器獲取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[‘a(chǎn)ccess_token‘]
        expires_in = tokenInfo[‘expires_in‘]
        self._writeWeixinTokenLog(token, self.order_no)
        return token
      except KeyError:
        return None #token獲取失敗
    return None #http請求失敗

獲取prepayid

在微信支付的開發(fā)流程中,最繁瑣的就是獲取prepayid。

這一步我們需要組裝這樣一個參數(shù):

{
"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&notify_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&notify_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需要的參數(shù)如上面代碼所示,所以我們需要準(zhǔn)備一個params,然后準(zhǔn)備簽名,簽名流程如下:

1.按照key的字典序,對params進(jìn)行排序,然后拼接成字符串,注意這些key不包括sign

2.在上面的字符串后面拼接key=paternerKey,然后對整個字符串進(jìn)行md5簽名,然后轉(zhuǎn)換成大寫,此時我們就得到了簽名

然后我們將所有params的value進(jìn)行urlencode轉(zhuǎn)碼,然后后面拼接上sign=signValue,就得到了package字符串。

這里創(chuàng)建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]])
    #加上財付通商戶權(quán)限密鑰
    stringSignTemp += ‘&key=%s‘ % (self.partnerKey)
    #使用MD5進(jìn)行簽名,然后轉(zhuǎn)化為大寫
    stringSign = hashlib.md5(stringSignTemp).hexdigest().upper()  #Upper
    return stringSign

組裝package的代碼:

def getPackage(self, packageParams):
    ‘‘‘先獲取params的sign,然后將params進(jìn)行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

繼續(xù)組裝參數(shù)

得到package后,我們繼續(xù)組裝參數(shù):

這里需要的參數(shù)為:

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&notify_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


注意這里有個坑:

參與簽名的是上面的參數(shù),但是最后的參數(shù)中不包括appKey,簽名后要記得刪除。

1.所有參數(shù)按照字典序排序,然后拼接

2.進(jìn)行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

隨后我們獲取到這樣的參數(shù):

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

獲取prepayid

代碼如下:

‘‘‘獲取預(yù)支付prepayid‘‘‘
  def gerPrepayId(self, token, requestParams):
    ‘‘‘將參數(shù),包括package,進(jìn)行json化,然后發(fā)起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格式應(yīng)該是這樣:

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

再次簽名

這里采用上面sha1的簽名方式再次簽名,獲取到下面的參數(shù):

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

后臺服務(wù)器將該結(jié)果返回給app,此時app即可發(fā)起支付。

上面的流程代碼為:

‘‘‘接收app的請求,返回prepayid‘‘‘
class WeixinRequirePrePaidHandler(BasicTemplateHandler):

  ‘‘‘這個方法在OrdersAddHandler中被調(diào)用‘‘‘
  @staticmethod
  def getPrePaidResult(order_no, total_pay, product_name, client_ip):
    ‘‘‘封裝了常用的簽名算法‘‘‘
    weixinRequestHandler = WeixinRequestHandler(order_no)
    ‘‘‘收集訂單相關(guān)信息‘‘‘
    addtion = str(random.randint(10, 100)) #產(chǎn)生一個兩位的數(shù)字,拼接在訂單號的后面
    out_trade_no = str(order_no) + addtion
    order_price = float(total_pay) #這里必須允許浮點數(shù),后面轉(zhuǎn)化成分之后轉(zhuǎn)化為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)

    ‘‘‘這里的一些參數(shù)供下面使用‘‘‘
    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:
      ‘‘‘設(shè)置package參數(shù)‘‘‘
      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)

      ‘‘‘設(shè)置支付參數(shù)‘‘‘
      signParams = {}
      signParams[‘a(chǎn)ppid‘] = config[‘a(chǎn)ppId‘]
      signParams[‘a(chǎn)ppkey‘] = config[‘paySignKey‘] #delete
      signParams[‘noncestr‘] = noncestr
      signParams[‘package‘] = package
      signParams[‘timestamp‘] = timestamp
      signParams[‘traceid‘] = ‘mytraceid_001‘

      ‘‘‘生成支付簽名‘‘‘
      app_signature = weixinRequestHandler.createSHA1Signature(signParams)
      ‘‘‘增加不參與簽名的額外參數(shù)‘‘‘
      signParams[‘sign_method‘] = ‘sha1‘
      signParams[‘a(chǎn)pp_signature‘] = app_signature

      ‘‘‘剔除appKey‘‘‘
      del signParams[‘a(chǎn)ppkey‘]

      ‘‘‘獲取prepayid‘‘‘
      prepayid = weixinRequestHandler.gerPrepayId(access_token, signParams)
      if prepayid:

        ‘‘‘使用拿到的prepayid再次準(zhǔn)備簽名‘‘‘
        pack = ‘sign=WXPay‘
        prepayParams = {}
        prepayParams[‘a(chǎn)ppid‘] = config[‘a(chǎn)ppId‘]
        prepayParams[‘a(chǎn)ppkey‘] = config[‘paySignKey‘]
        prepayParams[‘noncestr‘] = noncestr
        prepayParams[‘package‘] = pack
        prepayParams[‘partnerid‘] = config[‘partnerId‘]
        prepayParams[‘prepayid‘] = prepayid
        prepayParams[‘timestamp‘] = timestamp

        ‘‘‘生成簽名‘‘‘
        sign = weixinRequestHandler.createSHA1Signature(prepayParams)

        ‘‘‘準(zhǔn)備輸出參數(shù)‘‘‘
        returnParams = {}
        returnParams[‘status‘] = 0
        returnParams[‘retmsg‘] = ‘success‘
        returnParams[‘a(chǎn)ppid‘] = config[‘a(chǎn)ppId‘]
        returnParams[‘noncestr‘] = noncestr
        returnParams[‘package‘] = pack
        returnParams[‘prepayid‘] = prepayid
        returnParams[‘timestamp‘] = timestamp
        returnParams[‘sign‘] = sign
        returnParams[‘partnerId‘] = config[‘partnerId‘]
        returnParams[‘a(chǎn)ddtion‘] = addtion
        
      else:
        ‘‘‘prepayid獲取失敗‘‘‘
        returnParams = {}
        returnParams[‘status‘] = -1
        returnParams[‘retmsg‘] = ‘prepayid獲取失敗‘
    else: 
      ‘‘‘token獲取失敗‘‘‘
      returnParams = {}
      returnParams[‘status‘] = -1
      returnParams[‘retmsg‘] = ‘token獲取失敗‘

    ‘‘‘生成json格式文本,然后返回給APP‘‘‘
    return returnParams

后臺異步通知

微信服務(wù)器發(fā)來的notify異步通知,才是支付成功的最終標(biāo)志,這一步處于安全起見,我們必須進(jìn)行延簽:

延簽代碼如下:

def isTenpaySign(self, params):
    helper = WeixinRequestHandler()
    sign = helper.createMD5Signature(params)
    return params[‘sign‘] == sign

整體流程如下:

‘‘‘微信服務(wù)器向后臺發(fā)送的異步通知‘‘‘
class WeixinAppNotifyHandler(BasicTemplateHandler):
  def initialize(self):
    self.weixinResponseHandler = WeixinResponseHandler()

  def post(self):
    ‘‘‘解析參數(shù)‘‘‘
    params = self.parseQueryString()

    ‘‘‘驗證是否是weixin服務(wù)器發(fā)回的消息‘‘‘
    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) #更新訂單使用狀態(tài)
      consumeCouponByOrderNo(order_no) #優(yōu)惠券已經(jīng)使用
      self.write("success")
    else:
      self.write("fail")

  def parseQueryString(self):
    ‘‘‘獲取url中所有的參數(shù)‘‘‘
    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

最后說明一點,用戶在手機(jī)上付完款,并不算支付成功,只有weixin服務(wù)器收到notify通知返回的success時,才算交易最終成功,此時我們的手機(jī)可以收到微信官方發(fā)來的一條消息。

感謝各位的閱讀!關(guān)于“微信支付開發(fā)流程的示例分析”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,讓大家可以學(xué)到更多知識,如果覺得文章不錯,可以把它分享出去讓更多的人看到吧!

網(wǎng)站題目:微信支付開發(fā)流程的示例分析-創(chuàng)新互聯(lián)
文章源于:http://jinyejixie.com/article6/ccejig.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供品牌網(wǎng)站設(shè)計、建站公司網(wǎng)站設(shè)計公司、網(wǎng)站營銷、動態(tài)網(wǎng)站關(guān)鍵詞優(yōu)化

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

h5響應(yīng)式網(wǎng)站建設(shè)
广德县| 安化县| 德州市| 新巴尔虎左旗| 庄河市| 仙居县| 酒泉市| 洛川县| 五大连池市| 大城县| 巴彦县| 梁山县| 正阳县| 淅川县| 崇义县| 平度市| 平远县| 铜鼓县| 阜阳市| 治县。| 桦南县| 星座| 达日县| 资阳市| 瑞安市| 东光县| 蒙山县| 左权县| 荣昌县| 礼泉县| 海安县| 天等县| 墨玉县| 泸州市| 阿拉尔市| 鄱阳县| 榕江县| 靖宇县| 平湖市| 永安市| 永吉县|