我不确定这是否是微信端的问题,或者他们是否改变了他们的API。我们有一个正确响应微信端点验证的端点(与 echostr 相呼应),并且每当我们收到新消息时,我们也会收到 POST 消息。
问题是我们的帐户收到消息时收到的 POST 消息中没有任何 XML 内容。它发送其他所有内容(签名、随机数等),但不发送消息 XML。
我们做错了什么吗?API有变化吗?我们没有正确地握手吗?
XML 数据以原始 HTTP 请求正文发送。
作为字节字符串的原始 HTTP 请求正文。这对于以不同于传统 HTML 表单的方式处理数据非常有用:二进制图像、XML 有效负载等。
这是从微信发送到我的 Django 服务器的请求:
GET:<QueryDict: {u'nonce': [u'886****76'], u'timestamp': [u'1440041636'], u'signature': [u'29cb245a0f9399*******33956c3e96c500c56']}>,
POST:<QueryDict: {}>,
request.POST 为空,这意味着它不是常规的表单数据。
这就是我在 Django 上处理来自微信的 POST 消息的方式:使用 request.read()。
@csrf_exempt
def weixin(request):
logger.debug(request)
token = #YOUR TOKEN
if not validate(request.GET['signature'],request.GET['timestamp'],request.GET['nonce'],token):
return Http404('')
if request.method == 'GET':
echo_str = request.GET['echostr']
if echo_str != None:
return HttpResponse(echo_str)
else:
return Http404('')
elif request.method == 'POST':
reply_str = reply(request.read())
return HttpResponse(reply_str)
return Http404('')
def validate(signature, timestamp, nonce, token):
if signature == None or timestamp == None or nonce == None or token == None:
return False
seq = sorted([token, timestamp, nonce])
logger.debug(seq)
tmp_str = ''.join(seq)
encode_str = hashlib.sha1(tmp_str).hexdigest()
logger.debug(encode_str)
if signature == encode_str:
return True
else:
return False
我尚未验证的 PHP 代码。从 $GLOBALS["HTTP_RAW_POST_DATA"] 获取数据。
public function responseMsg()
{
$postStr = $GLOBALS["HTTP_RAW_POST_DATA"];
if (!empty($postStr)){
$postObj = simplexml_load_string($postStr, 'SimpleXMLElement', LIBXML_NOCDATA);
$fromUsername = $postObj->FromUserName;
$toUsername = $postObj->ToUserName;
$keyword = trim($postObj->Content);
$time = time();
$textTpl = "<xml>
<ToUserName><![CDATA[%s]]></ToUserName>
<FromUserName><![CDATA[%s]]></FromUserName>
<CreateTime>%s</CreateTime>
<MsgType><![CDATA[%s]]></MsgType>
<Content><![CDATA[%s]]></Content>
<FuncFlag>0</FuncFlag>
</xml>";
if($keyword == "?" || $keyword == "?")
{
$msgType = "text";
$contentStr = date("Y-m-d H:i:s",time());
$resultStr = sprintf($textTpl, $fromUsername, $toUsername, $time, $msgType, $contentStr);
echo $resultStr;
}
}else{
echo "";
exit;
}
}