正确填写与获取下载文件名

2021-06-18 12:23:46 +08:00
 anzu

我用 requests 下载文件,文件名在 response header 的 Content-Disposition 中, 在浏览器中虽然 devtools 解析乱码,但下载弹框的文件名正确,而在程序中获取到的内容也是乱码,类似于

attachment; filename=影宅 シャドーハウス 01;filename*=utf-8''影宅 シャドーハウス 01

经过尝试,可能是进行了如下错误解码

    raw = '影宅 シャドーハウス 01'
    b = bytes(raw, 'utf8')
    dec = b.decode(encoding='iso-8859-1', errors='replace')
    print(dec)

output:

影宅 シャドーハウス 01

逆向解码:

print(bytes(dec, 'iso-8859-1').decode('utf8'))

按规范来说,接收方不保证能解析 utf8 header,所以各位开发同学请不要再在 header 中填入 utf8 内容了!

2367 次点击
所在节点    Python
4 条回复
swulling
2021-06-18 13:03:25 +08:00
嗯,放 utf8 不符合规范,标准做法是 URL-encode
Kobayashi
2021-06-18 14:11:39 +08:00
"attachment; filename*=utf-8''{}".format(quote(filename))

- filename* 而不是 filename
- utf-8 后 2 个单引号不是打字错误
- quote() 为 urllib.parse.quote

纯 ascii 字符使用 filename=,否则 filename*=

完整代码参考 Starlette FileResponse 对象

https://github.com/encode/starlette/blob/15761fb48e4c56be09167cb8f9b761114593b651/starlette/responses.py#L257-L265
Kobayashi
2021-06-18 14:17:39 +08:00
当然并不是所有人都按规矩办事。规范中 header 编码是 latin-1/iso-8859-1,对方服务器可能使用 utf-8 编码 header,而浏览器默认使用 iso-8859-1 解码有乱码。所以有了 encode('iso-8859-1').decode('utf-8').

以前在腾讯乐享的 Set-Cookie 见过这种情况。Chrome 旧版会尝试使用 utf-8 解码,新版遵循了规范只使用 iso-8859-1.
coffeeing
2022-11-07 15:15:46 +08:00
@Kobayashi "attachment; filename*=utf-8''{}".format(quote(filename))
打扰,麻烦问下,这个代码是写在 http 的 head 里,还是服务端的配置文件里?谢谢谢谢

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/784203

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX