Python请求以utf-8编码的响应,但无法解码

Taa*_*ako 6 python post request facebook-messenger

我正在尝试使用python抓取我的messenger.com(facebook Messenger)聊天,并且我已经使用Google chromes开发者工具查看聊天历史记录的POST请求,并且我已将整个标头和正文复制为请求可以使用的格式。

我得到的HTTP代码200暗示该请求至少有一些东西,但是我可以print res.encoding得到它返回的编码,其中说的是utf-8。但是我无法解码!

这是函数:

def download_thread(self, limit, offset, message_timestamp):
    """Download the specified number of messages from the
    provided thread, with an optional offset
    """
    data = request_data(self.thread, offset=offset,
                        limit=limit, group=self.group,
                        timestamp=message_timestamp)

    res = self.ses.post(url_thread, data=data, headers=headers)

    print(res.content)

    thread_contents = json.loads(res.content)
    print(thread_contents)
    return thread_contents
Run Code Online (Sandbox Code Playgroud)

产量

UnicodeDecodeError: 'utf-8' codec can't decode byte 0x87 in position 0: invalid start byte
Run Code Online (Sandbox Code Playgroud)

当它尝试json.load(或loads)数据时

但是res.encoding会返回utf-8。

我尝试用gzip解压缩,但是那不是gzip压缩的内容。

如果我只是试着做print(res.content)我得到

Traceback (most recent call last):
  File "FBChatScraper.py", line 200, in <module>
    main()
  File "FBChatScraper.py", line 134, in main
    fbms.run()
0f\x82\x048\xbb\xb9=\x87\xebK0.\xff\x90\xdd\xeb\xfa\x16\xc6\xbbz\x8b\x82)\xe8\xaaV\x01^\xda\x8b\xbd\x15d-\xb1\x10@\x17\\\xd43\xa8\x92w\xe8\xc0\xcdU\xc4\xff\xc7\xfa\x90\xb2\xb3\xf5\x84\x11u\x0b\t\x8f\x83r\xf3}\xe5!y$\xe6\xf6c0\xf0\xb4\x98\xcat_\x0c\x08\xb5\xdd\x8ctx\x91\xa9\x95\rB%\xe2\x93\xa52\x85_\xa6\x10\xc2\xc9\xa3\xee4SDb\xa5\x18QJ\x83X\x19)\xaa$\xf4\xb4\xb7\x0b\x84\x15&\x88\x08L\xc9iP\xa2\xb9\xf2\xaf\x96\x96N\xd8\xcf=\x05\xc1\x18\x8d\xa0\xf2Y\x8e\n\xcf\xc8\x0fE4\xd6)\xa1\xd4\xb7D\xd6{i\xc8P\x96R\x11HC\xac\xbcKyT#~}\x93\xf7@K\xc7r/\x82\xb0\xe4\xefX\xf9j\x08\xa6Hp\xfcn\x06\xfdo\x9a\xd0wJ\xb4fJ(\x89+\x1c\xf6\x0eOI\x90\xac\x9eDD\xfd,\xa5\xe9\x89\x1blh\x86Z\x98\x05\xdd9\xc7\xf4\x80\xfcY\x8e\xad\xee\x99!\x15\x13+\x9b\x07\xe8Fdj\xfc\x11\xfc\xfe7\x06h\x02\x00@>]W\x92\xc9\x02\xb1c3\x82\xcd\xa4\xefN9\x90\xe6\x81y\x9c\x84er\xd4\xc3\x06\x1c\x06\x14\xcf\xc7\x07hj\xbfH\xdc\xf5~\xf7z\x18Ce\xaf^\x8c\xab \xdfV\xce\xb8\x11\xf8\x06\x03'

Traceback (most recent call last):
  File "FBChatScraper.py", line 200, in <module>
    main()
  File "FBChatScraper.py", line 134, in main
    fbms.run()
  File "FBChatScraper.py", line 43, in run
    thread_contents = self.download_thread(limit, offset, message_timestamp)
  File "FBChatScraper.py", line 74, in download_thread
    thread_contents = json.loads(res.content)
  File "/Users/silman/anaconda/lib/python3.6/json/__init__.py", line 349, in loads
    s = s.decode(detect_encoding(s), 'surrogatepass')
UnicodeDecodeError: 'utf-8' codec can't decode byte 0x87 in position 0: invalid start byte
Run Code Online (Sandbox Code Playgroud)

在回溯的中间奇怪地打印内容,使我认为有一些看不见的字符将其压低。

我无法将响应加载为json格式,因为无论我如何处理响应内容,其格式都不适合json库解释。

此外,如果我只是做print(res.text)我会垃圾:

Traceback (most recent call last):
  File "FBChatScraper.py", line 200, in <module>
    main()
  File "FBChatScraper.py", line 134, in main
    fbms.run()
}sP???c???f?u0???\? QZed?C??? M$x???H?????e??]???5???^?*??aM?Y??b???/??JW/???>H6z?\??l4????t=i??%?u?x??%?x?
       F    <???{1i?#%;?r?=R?m??1B?Z(+?(S-???#??\v?{b??
                                                           ?    f/V?i???_??83?  ?_????*??O??
                                                                                            ??????Z??i-?TVeaG54?!v?a??|gu-g??.???"J$?L`&?t?#s)?H????s???q???^?0??[)???j???T???U???J??wW???!eg?#j ??r??$y???3?4??4.??M?@Kb?AX?SDb?QJ?X)?,???a?   "Sp?h?????sOA0Vé|???????:%?rKdKC???@ M??.?^
?       ?g???SWQH?.??B?G?,????@E????????
                                        nras??L?/??ch@>]W???c3???N9??y??er????hj?H??~?zCe?^?? ?V??

Traceback (most recent call last):
  File "FBChatScraper.py", line 200, in <module>
    main()
  File "FBChatScraper.py", line 134, in main
    fbms.run()
  File "FBChatScraper.py", line 43, in run
    thread_contents = self.download_thread(limit, offset, message_timestamp)
  File "FBChatScraper.py", line 74, in download_thread
    thread_contents = json.loads(res.content)
  File "/Users/silman/anaconda/lib/python3.6/json/__init__.py", line 349, in loads
    s = s.decode(detect_encoding(s), 'surrogatepass')
UnicodeDecodeError: 'utf-8' codec can't decode byte 0x87 in position 0: invalid start byte
Run Code Online (Sandbox Code Playgroud)

编辑:

最好的MWE,我不确定发帖请求中的哪些数据是私人的,所以我省略了一些

使用此数据

url_thread = "https://www.messenger.com/api/graphqlbatch/"


request_data = {
  "batch_name": "MessengerGraphQLThreadFetcher",
  "__user": "<user_id>",
  "__a": "1",
  "__dyn": "<dyn>",
  "__req": "9",
  '__be'      : '-1',
  '__pc'      : 'PHASED:messengerdotcom_pkg',
  "fb_dtsg": "AQFni7TU2nes:AQGSC8FSDqyw",
  "ttstamp": "265817254666710077746711957586581715370521181008510710777",
  "__rev": "3791607",
  "jazoest": "<jazoest>",
  "queries": '<queries>'
  }

headers = {
  "authority": "www.messenger.com",
  "method": "POST",
  "path": "/api/graphqlbatch/",
  "scheme": "https",
  "accept": "*/*",
  "accept-encoding": "gzip, deflate, br",
  "accept-language": "en-US,en;q=0.9",
  "cache-control": "no-cache",
  "content-length": "754",
  "content-type" : "application/x-www-form-urlencoded",
  "cookie": "<cookies>",
  "origin": "https://www.messenger.com",
  "pragma": "no-cache",
  "referer": "https://www.messenger.com/t/<chatID>",
  "user-agent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_13_3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/65.0.3325.181 Safari/537.36"
}
Run Code Online (Sandbox Code Playgroud)

您可以<items>使用chrome开发人员工具并在“网络”标签上查找,以获取POST请求Request URL: https://www.messenger.com/api/graphqlbatch/

如果您在记录chrome开发工具时向上滚动以重新加载旧消息,则很容易找到。

然后用python组合一个简单的请求

import requests as rq
import time

ses = rq.Session()
thread = <ID of thread found in URL of messenger.com>

conversation_type = <'thread_fbids' if group chat else 'user_ids'>

data = request_data
data['messages[{}][{}][offset]'.format(conversation_type, thread)] = 0
data['messages[{}][{}][timestamp]'.format(conversation_type, thread)] = int(time.time())
data['messages[{}][{}][limit]'.format(conversation_type, thread)] = 2000

res = ses.post(url_thread, data=data, headers=headers)

print(res.content)
thread_contents = json.loads(res.content)
print(thread_contents)
Run Code Online (Sandbox Code Playgroud)

当我的开发工具返回时,您可以在此处看到j​​son的开头

aba*_*ert 7

问题是您的请求标头中的以下行:

"accept-encoding": "gzip, deflate, br",
Run Code Online (Sandbox Code Playgroud)

br需要Brotli压缩,这是Google试图取代网络上的gzip的一种新的压缩标准(请参阅RFC 7932)。Chrome之所以要求Brotli是因为最新版本的Chrome本身就可以理解。您之所以要Brotli,是因为您是从Chrome复制标题的。但是requests并不了解Brotli。

您可以pip install brotli注册解压缩器,也可以直接在上手动调用它res.content。但是更简单的解决方案是删除br

"accept-encoding": "gzip, deflate",
Run Code Online (Sandbox Code Playgroud)

…然后您应该得到gzip,并且您requests已经知道如何处理。

  • 这就是我从开发工具中盲目复制标头得到的结果:p,谢谢! (2认同)