通知機制的實現,官方只有文檔沒有demo代碼,對沒搞過的人來說,需要花大量時間來做測試。
從文檔上說的來看,微信每次通知過來的數據,結構比較復雜,是一個多段數據,除了要取出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部分。這也幫助我們理解了,為什麼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之外為空外,其它情況都可能不為空
以上轉述這麼多文字的意思,就是說,得用到這兩種方式來讀取微信傳過來的數據。
先取$POST 這是常規的支付通知信息,形如:
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')讀取額外的信息,形如:
<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>
最後,做相應的業務邏輯處理,就不詳述了。