1

我不确定这是否是微信端的问题,或者他们是否改变了他们的API。我们有一个正确响应微信端点验证的端点(与 echostr 相呼应),并且每当我们收到新消息时,我们也会收到 POST 消息。

问题是我们的帐户收到消息时收到的 POST 消息中没有任何 XML 内容。它发送其他所有内容(签名、随机数等),但不发送消息 XML。

我们做错了什么吗?API有变化吗?我们没有正确地握手吗?

4

1 回答 1

1

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;
        }
    }
于 2015-08-20T03:48:57.647 回答