python3中用django下载文件,中文名乱码怎么办?

前段时间被某个前端小可爱鄙视了一下,说我博客都一年不更新了,我不服,明明还有俩月才到一年呢。不过说是这么说,还是要更新一下的。
以上都是借口,下面开始正文。
 
 
我公司的某个内部系统,用django做的,项目中不可避免地有下载文件的地方,以前偷懒,我都是用django自带的方法,在项目的总urls.py中使用
 
urlpatterns += static(FILEPATH, document_root=FILEPATH)

 

这种方法解决。
 
但是这种方法有个极大的缺陷:测试环境写着玩可以,正式环境肯定要把settings中的debug=True关掉的。而这种方法,在关掉debug之后,就不能用了。
 
于是我只好走传统线路:设置header。
 
在urls.py中增加:
 
url(r"^download/$", views.download, name="download")

 

在页面中:
 
<a href="{% url 'download' %}">下载</a>

 

在views.py中增加一个视图函数:
 
def download(request):
    return build_download_response(FILE_PATH, "我爱可乐.docx")

 

(MTV模式的程咬金三板斧,没毛病)
 
由于系统里面不止一个地方用到下载功能,所以,我把它写成了一个通用的函数:
 
from django.http import FileResponse
 
def build_download_response(filepath, filename):
    """
    构建下载文件的文件头
    :param filepath: 文件路径
    :param filename: 文件名
    :return: FileResponse
    """
    absname = os.path.join(filepath, filename) if os.path.isdir(filepath) else filepath
 
    response = FileResponse(open(absname, "rb"))
    response["Content-Type"] = "application/octet-stream"
    response["Content-Disposition"] = "attachment; filename='%s'" % filename
    return response
 

 

参照网上的说法,这样是没问题的。蓝鹅,它确实出了问题:当我点击下载按钮的时候,弹出的界面只有“下载”二字。
 
 
喵喵喵?我的文件呢?
我尝试把文件下载下来看了一下,从大小来看,是没有问题的。
 
 
我把名字改过来,可以正常打开,里面没有丢东西——换句话说,只是文件名出了问题
可是文件名能有毛的问题啊?你不能欺负我公司的人只会中文吧!
 
 
 
经过一番激烈地百度和扣人心弦地搜索,我从这篇博客 https://blog.csdn.net/u011090495/article/details/18815777 里找到了原因。
简单点说就是,Content-Disposition里面的filename这个东西,原来不是RFC标准,仅支持ASCII编码的文件名。如果文件名不是英文的,恐怕就会出现名字乱码,或者被改名的情况那么怎么办呢?另一个RFC给加上了扩展,可以定义一个filename*,然后把文件名编码一下。
于是,我把 Content-Disposition 这里从
 
response["Content-Disposition"] = "attachment; filename='%s'" % filename

 

改成了
 
response["Content-Disposition"] = "attachment; filename='%s'; filename*=UTF-8''%s" % (filename.encode("UTF-8"), filename.encode("UTF-8"))

 

结果测试了一下,哭了。
 
 
还真是编码了……
我要闹了。
 
还好,后来又在 https://segmentfault.com/q/1010000009078463 这篇帖子里找到了答案。原来 filename*=UTF-8''%s" % filename.encode("UTF-8") 这样写是不管用的,django自有django的方法:
 
from django.http import FileResponse
from django.utils.encoding import escape_uri_path
 
def build_download_response(filepath, filename):
    """
    构建下载文件的文件头
    :param filepath: 文件路径
    :param filename: 文件名
    :return: FileResponse
    """
    absname = os.path.join(filepath, filename) if os.path.isdir(filepath) else filepath
 
    response = FileResponse(open(absname, "rb"))
    response["Content-Type"] = "application/octet-stream"
    response["Content-Disposition"] = "attachment; filename*=UTF-8''{}".format(escape_uri_path(filename))
 
    return response
 

 

于是,就可以正常下载了。
 
 
 
打开看看,内容完全没问题!
 
 
 
开心!(我才不会拿去给前端炫耀呢!)
 
posted @ 2019-08-29 18:07  _小苹果  阅读(2994)  评论(1编辑  收藏  举报