微信支付通知的處理方式簡要解析

通知機制的實現,官方只有文檔沒有demo代碼,對沒搞過的人來講,須要花大量時間來作測試。php

 

從文檔上說的來看,微信每次通知過來的數據,結構比較複雜,是一個多段數據,除了要取出POST數據外,還要取其它的數據。數組

 

這裏首先涉及到一個關於php://input與$_POST取值的問題,簡單列幾點以下:微信

 

1,Content- Type取值爲application/x-www-form-urlencoded時,php會將http請求body相應數據會填入到數組$_POST,填入到$_POST數組中的數據是進行urldecode()解析的結果。(其實,除了該Content-Type,還有 multipart/form-data表示數據是表單數據,稍後咱們介紹)
2,php://input數據,只要Content-Type不爲 multipart/form-data(該條件限制稍後會介紹)。那麼php://input數據與http entity body部分數據是一致的。該部分相一致的數據的長度由Content-Length指定。
3,僅當Content-Type爲application/x-www-form-urlencoded且提交方法是POST方法時,$_POST數據與php://input數據纔是」一致」(打上引號,表示它們格式不一致,內容一致)的。其它狀況,它們都不一致。
4,php://input讀取不到$_GET數據。是由於$_GET數據做爲query_path寫在http請求頭部(header)的PATH字段,而不是寫在http請求的body部分。app

這也幫助咱們理解了,爲何xml_rpc服務端讀取數據都是經過file_get_contents(‘php://input', ‘r')。而不是從$_POST中讀取,正是由於xml_rpc數據規格是xml,它的Content-Type是text/xml。
5. php://input碰到了multipart/form-data,請查閱RFC1867對它的描述。multipart/form-data也表示以POST方法提交表單數據,它還伴隨了文件上傳,因此會跟application/x- www-form-urlencoded數據格式不同。它會以一更種更合理的,更高效的數據格式傳遞給服務端。當Content-Type爲multipart/form-data的時候,即使http請求body中存在數據,php://input也爲空,PHP此時,不會把數據填入php://input流。因此,能夠肯定: php://input不能用於讀取enctype=multipart/form-data數據。測試

6. 當Content-Type爲application/x- www-form-urlencoded時,php://input和$_POST數據是「一致」的,爲其它Content-Type的時候,php: //input和$_POST數據數據是不一致的。由於只有在Content-Type爲application/x-www-form- urlencoded或者爲multipart/form-data的時候,PHP纔會將http請求數據包中的body相應部分數據填入$_POST全局變量中,其它狀況PHP都忽略。而php://input除了在數據類型爲multipart/form-data以外爲空外,其它狀況均可能不爲空url

以上轉述這麼多文字的意思,就是說,得用到這兩種方式來讀取微信傳過來的數據。
spa

先取$POST 這是常規的支付通知信息,形如:code

 

array (
  'bank_type' => '3006',
  'discount' => '0',
  'fee_type' => '1',
  'input_charset' => 'UTF-8',
  'notify_id' => 'YaNO6cznoNZK0aGb8nJWGgVUWssjt7Ze7gWRaRS0R_5w9oXgGNkRGxReEk0r45yk3I9a2_gzo9IqgqMYbap6bxC2T3p0o-2C',
  'out_trade_no' => '1214284731',
  'partner' => '12xxxxxxxx',
  'product_fee' => '3400',
  'sign' => '545FA0E8B594BBXXXX48XX142F084TY',
  'sign_type' => 'MD5',
  'time_end' => '20130223110224',
  'total_fee' => '3400',
  'trade_mode' => '1',
  'trade_state' => '0',
  'transaction_id' => '12XXX449012014XXX33174005XXX',
  'transport_fee' => '0',
)

 

再用file_get_contents('php://input')讀取額外的信息,形如:orm

 

<xml><OpenId><![CDATA[o0pd3jqHaN7b0tVPDFJPzJEkSCLw]]></OpenId>
<AppId><![CDATA[wxXXX06XX2cXXX88XX]]></AppId>
<IsSubscribe>1</IsSubscribe>
<TimeStamp>1400814743</TimeStamp>
<NonceStr><![CDATA[lqxwMsiY9EXRDpms]]></NonceStr>
<AppSignature><![CDATA[c2dxxxe186116b32b06axxxc1a688b671eexxx5e]]></AppSignature>
<SignMethod><![CDATA[sha1]]></SignMethod>
</xml>

 

最後,作相應的業務邏輯處理,就不詳述了。xml

相關文章
相關標籤/搜索